1
0
mirror of https://github.com/LDDestroier/CC/ synced 2025-01-22 06:56:53 +00:00
ldd-CC/progdor2.lua
2021-07-03 07:49:23 -04:00

1235 lines
32 KiB
Lua

--[[
Progdor 2.0 - File Packaging
by LDDestroier
Get with:
wget https://raw.githubusercontent.com/LDDestroier/CC/master/progdor2.lua
Uses CCA compression API, made by minizbot2012.
--]]
local progdor = {
version = "2.0",
PBlogPath = ".progdor_PB_uploads",
channel = 8366,
skynetPath = "skynet.lua",
skynetURL = "https://github.com/osmarks/skynet/raw/master/client.lua"
}
local scr_x, scr_y = term.getSize()
local modem = peripheral.find("modem")
local skynet, skynetBigReceive, skynetBigSend
local function interpretArgs(tInput, tArgs)
local output = {}
local errors = {}
local usedEntries = {}
for aName, aType in pairs(tArgs) do
output[aName] = false
for i = 1, #tInput do
if not usedEntries[i] then
if tInput[i] == aName and not output[aName] then
if aType then
usedEntries[i] = true
if type(tInput[i+1]) == aType or type(tonumber(tInput[i+1])) == aType then
usedEntries[i+1] = true
if aType == "number" then
output[aName] = tonumber(tInput[i+1])
else
output[aName] = tInput[i+1]
end
else
output[aName] = nil
errors[1] = errors[1] and (errors[1] + 1) or 1
errors[aName] = "expected " .. aType .. ", got " .. type(tInput[i+1])
end
else
usedEntries[i] = true
output[aName] = true
end
end
end
end
end
for i = 1, #tInput do
if not usedEntries[i] then
output[#output+1] = tInput[i]
end
end
return output, errors
end
local yield = function()
os.queueEvent("yield")
os.pullEvent("yield")
end
-- CCA API START --
local bit = bit32
local function pack(bn1, bn2)
return bit.band(bn1, 0xFF), bit.rshift(bn1, 8) + bit.lshift(bit.band(bn2, 0xF), 4), bit.rshift(bn2, 4)
end
local function upack(b1, b2, b3)
return (b1 + bit.lshift(bit.band(b2, 0xF), 8)), (bit.lshift(b3,4) + bit.band(bit.rshift(b2, 4), 0xF))
end
local function createDict(bool)
local ret = {}
for i = 1, 255 do
if bool then
ret[string.char(i)] = i
else
ret[i] = string.char(i)
end
end
if not bool then ret[256] = 256 end
return ret
end
local function cp(sInput)
local dic = createDict(true)
local s = ""
local ch
local dlen = 256
local result = {}
local temp
for i = 1, #sInput do
if dlen == 4095 then
result[#result + 1] = dic[s]
result[#result + 1] = 256
dic = createDict(true)
dlen = 256
s = ""
end
ch = sInput:sub(i, i)
temp = s..ch
if dic[temp] then
s = temp
else
result[#result + 1] = dic[s]
dlen = dlen +1
dic[temp] = dlen
s = ch
end
end
result[#result + 1] = dic[s]
return result
end
local function dc(data)
local dic = createDict(false)
local entry
local ch
local currCode
local result = {}
result[#result + 1] = dic[data[1]]
prefix = dic[data[1]]
for i = 2, #data do
currCode = data[i]
if currCode == 256 then
dic = createDict(false)
prefix = ""
else
entry = dic[currCode]
if entry then -- exists in dictionary
ch = entry:sub(1, 1)
result[#result + 1] = entry
if prefix ~= "" then
dic[#dic+1] = prefix .. ch
end
else
ch = prefix:sub(1, 1)
result[#result + 1] = prefix..ch
dic[#dic + 1] = prefix..ch
end
prefix = dic[currCode]
end
end
return table.concat(result)
end
local function trim(inp)
for i = 0,2 do
if inp[#inp] == 0 then
inp[#inp] = nil
end
end
end
local function decompress(input)
local rec = {}
for i = 1, #input, 3 do
if i % 66 == 0 then
yield()
end
rec[#rec+1], rec[#rec+2] = upack(input[i], input[i+1] or 0, input[i+2] or 0)
end
trim(rec)
return dc(rec)
end
local function compress(input)
local rec = {}
local data = cp(input)
for i=1, #data, 2 do
yield()
rec[#rec+1], rec[#rec+2], rec[#rec+3] = pack(data[i], data[i+1] or 0)
end
trim(rec)
return rec
end
local function strCompress(input)
local output = {}
local tbl = compress(input)
for i = 1, #tbl do
output[i] = string.char(tbl[i])
end
return table.concat(output)
end
local function strDecompress(input)
local output = {}
for i = 1, #input do
output[i] = string.byte(input:sub(i,i))
end
return decompress(output)
end
-- CCA API END --
-- colors that are always safe to set to
local safeColorList = {
[colors.white] = true,
[colors.lightGray] = true,
[colors.gray] = true,
[colors.black] = true
}
-- pastebin uploads have a 512K limit
local pastebinFileSizeLimit = 1024 * 512
local argData = {
["-pb"] = "string", -- pastebin get
["-dd"] = "string", -- direct URL download
["-m"] = "string", -- specify main file
["-PB"] = false, -- pastebin upload
["-t"] = false, -- transmit file
["-r"] = false, -- receive file
["-R"] = false, -- include read-only files
["-P"] = false, -- include Progdor2 file
["-S"] = false, -- use skynet
["-e"] = false, -- automatic self-extractor
["-E"] = "string", -- specify output folder in self-extractor code
["-s"] = false, -- silent
["-a"] = false, -- use as API with require, also makes silent
["-c"] = false, -- use CCA compression
["-h"] = false, -- show help
["-i"] = false, -- inspect mode
["-o"] = false, -- always overwrite
}
local argList, argErrors = interpretArgs({...}, argData)
if #argErrors > 0 then
local errList = ""
for k,v in pairs(argErrors) do
if k ~= 1 then
errList = errList .. "\"" .. k .. "\": " .. v .. "; "
end
error(errList:sub(1, -2))
end
end
local pastebinGet = argList["-pb"] -- string, pastebin code
local directDownload = argList["-dd"] -- string, download URL
local mainFile = argList["-m"] -- string, main executable file
local pastebinUpload = argList["-PB"] -- boolean
local selfExtractor = argList["-e"] -- boolean
local selfExtractorFolder = argList["-E"] -- string, folder output for self extractor code
local silent = argList["-s"] -- boolean
local useCompression = argList["-c"] -- boolean
local justOverwrite = argList["-o"] -- boolean
local allowReadOnly = argList["-R"] -- boolean
local allowPackPD = argList["-P"] -- boolean
local useSkynet = argList["-S"] -- boolean
local trMode = argList["-t"] and "transmit" or (argList["-r"] and "receive" or "normal")
local skynet
if useCompression and selfExtract then
error("Cannot use compression with self-extractor.")
end
local sWrite = function(text)
if not silent then
return write(text)
end
end
local sPrint = function(text)
if not silent then
return print(text)
end
end
local cWrite = function(text, color, ignoreSilent)
local col = term.getTextColor()
term.setTextColor(color or col)
if ignoreSilent then
write(text)
else
sWrite(text)
end
term.setTextColor(col)
end
local cPrint = function(text, color, ignoreSilent)
local col = term.getTextColor()
term.setTextColor(color or col)
if ignoreSilent then
print(text)
else
sPrint(text)
end
term.setTextColor(col)
end
local function showHelp(verboseHelp)
local helpInfo
if verboseHelp then
helpInfo = {
"Progdor v" .. progdor.version,
" -pb [pastebin ID] : Download from Pastebin.", -- added
" -PB : Upload to pastebin.", -- added
" -dd [download URL] : Download from URL.", -- added
" -e : Adds on self-extractor code to archive.", -- added
" -E [folder] : Extractor extracts to folder", -- added
" -s : Silences all terminal writing", -- added
" -S : Use skynet when transmitting/receiving.", -- added
" -t : Transmit a folder/file.", -- added
" -r : Receive a file/packed folder.", -- added
" -R : Allow packing read-only files/folders.", -- added
" -P : Allow packing in Progdor2 itself.", -- added
" -a : Allows programs to use require() on Progdor.", -- added
" -c : Enables CCA compression.", -- added
" -m : Specify main executable file in archive.", -- added
" -i : Inspect archive without extracting.", -- added
" -o : Overwrite files without asking.", -- added
}
else
helpInfo = {
"Progdor v" .. progdor.version,
"Usage: progdor [options] inputFolder (outputFile)",
" progdor [options] inputFile (outputFolder)",
"",
"Progdor is a file/folder packaging program with support for CCA compression and self-extraction.",
" If tacking on auto-extractor, a third argument will be the default extraction path.",
"",
"Use -h for all options.",
"",
" This Progdor has Super Cow Powers." -- not actually added
}
end
for y = 1, #helpInfo do
sPrint(helpInfo[y])
end
end
local setTextColor = function(color)
if (not silent) and (term.isColor() or safeColorList[color]) then
term.setTextColor(color)
end
end
local setBackgroundColor = function(color)
if (not silent) and (term.isColor() or safeColorList[color]) then
term.setBackgroundColor(color)
end
end
local inputPath = argList[1]
local outputPath = argList[2] or inputPath
local defaultAutoExtractPath = argList[3]
local exists, mode
if inputPath == "moo" and not fs.exists(inputPath) then
print([[
\_/
m00h (__) -(_)-
\ ~Oo~___ / \
(..) |\
___________|_|_|_____________
..."Have you mooed today?"..."]])
return
end
if argList["-h"] then
return showHelp(true)
elseif argList["-a"] or (not shell) then
mode = "api"
elseif inputPath then
exists = fs.exists(inputPath)
if argList["-i"] then
mode = "inspect"
elseif fs.isDir(inputPath) then
mode = "pack"
else
mode = "unpack"
end
elseif trMode ~= "receive" then
return showHelp(false)
end
if mode == "api" then
silent = true
elseif (pastebinGet or directDownload) and pastebinUpload then
error("Cannot upload and download at the same time!")
end
local specialWrite = function(left, colored, right, color)
local origTextColor = term.getTextColor()
sWrite(left)
setTextColor(color)
sWrite(colored)
setTextColor(origTextColor)
sWrite(right)
end
local specialPrint = function(left, colored, right, color)
return specialWrite(left, colored, right .. "\n", color)
end
local function listAll(path, includePath)
local output = {}
local list = fs.list(path)
local fc = fs.combine
for i = 1, #list do
if allowReadOnly or (not fs.isReadOnly(fc(path, list[i]))) then
if allowPackPD or fc(path, list[i]) ~= (shell and shell.getRunningProgram()) then
if fs.isDir(fc(path, list[i])) then
if #fs.list(fc(path, list[i])) == 0 then
output[#output+1] = (includePath and fc(path, list[i]) or list[i]) .. "/"
else
local la = listAll(fc(path, list[i]))
for ii = 1, #la do
output[#output+1] = includePath and fc(path, fc(list[i], la[ii])) or fc(list[i], la[ii])
end
end
else
output[#output+1] = includePath and fc(path, list[i]) or list[i]
end
end
end
end
return output
end
local makeFileList = function(path, doCompress)
local output = {}
local list = listAll(path, false)
local file
if not allowPackPD then
cPrint("Ignoring Progdor2.", colors.lightGray)
end
if not allowReadOnly then
cPrint("Ignoring read-only files.", colors.lightGray)
end
sPrint("Packing files...")
for i = 1, #list do
setTextColor(colors.lightGray)
sWrite("'" .. list[i] .. "'...")
if list[i]:sub(-1,-1) == "/" then
output[list[i]] = true -- indicates empty directory
else
file = fs.open(fs.combine(path, list[i]), "r")
output[list[i]] = doCompress and strCompress(file.readAll()) or file.readAll()
file.close()
setTextColor(colors.green)
sPrint("good")
end
end
setTextColor(colors.white)
return output
end
local buildArchive = function(path, mainFile, doCompress)
local output = {
compressed = doCompress, -- uses CCA compression
mainFile = mainFile, -- specifies the main program within the archive to run, should I implement something to use that
data = makeFileList(path, doCompress) -- files and folders and whatnot
}
return textutils.serialize(output)
end
local parseArchiveData = function(input, doNotDecompress)
local archive = textutils.unserialize(input)
if archive then
if archive.compressed and (not doNotDecompress) then
for name, contents in pairs(archive.data) do
archive.data[name] = strDecompress(contents)
end
archive.compressed = false
end
return archive
else
return false
end
end
local parseArchive = function(path, doNotDecompress)
local file = fs.open(path, "r")
local output = parseArchiveData(file.readAll(), doNotDecompress)
file.close()
return output
end
local round = function(number, places)
return math.floor(number * (10^places)) / (10^places)
end
local choice = function(input,verbose)
if not input then
input = "yn"
end
if verbose then
sWrite("[")
for a = 1, #input do
sWrite(input:sub(a,a):upper())
if a < #input then
sWrite(",")
end
end
sWrite("]?")
end
local evt,char
repeat
evt,char = os.pullEvent("char")
until string.find(input:lower(),char:lower())
if verbose then
sPrint(char:upper())
end
local pos = string.find(input:lower(), char:lower())
return pos, char:lower()
end
local overwriteOutputPath = function(inputPath, outputPath, allowMerge, override)
setTextColor(colors.white)
local c
if override then
return true, true
else
if allowMerge then
write("Overwrite [Y/N]? Or [M]erge? ")
c = choice("ynm", false)
else
write("Overwrite [Y/N]?")
c = choice("yn", false)
end
write("\n")
if c == 1 then
return true, true
elseif c == 2 then
sPrint("Abort.")
return false, false
elseif c == 3 then
return true, false
end
end
end
local uploadToPastebin = function(archive, name)
if #archive > pastebinFileSizeLimit then
error("That archive is too large to be uploaded to Pastebin. (limit is 512 KB)")
return false
else
local key = "0ec2eb25b6166c0c27a394ae118ad829"
local response = http.post(
"https://pastebin.com/api/api_post.php",
"api_option=paste&" ..
"api_dev_key=" .. key .. "&" ..
"api_paste_format=lua&" ..
"api_paste_name=" .. textutils.urlEncode(name) .. "&" ..
"api_paste_code=" .. textutils.urlEncode(archive)
)
if response then
local sResponse = response.readAll()
response.close()
local sCode = string.match( sResponse, "[^/]+$" )
return sCode, sResponse
else
return false
end
end
end
local writeArchiveData = function(archive, outputPath)
local file
for name, contents in pairs(archive.data) do
setTextColor(colors.lightGray)
sWrite("'" .. name .. "'...")
if contents == true then -- indicates empty directory
fs.makeDir(fs.combine(outputPath, name))
else
file = fs.open(fs.combine(outputPath, name), "w")
if file then
file.write(contents)
file.close()
end
end
if file then
setTextColor(colors.green)
sPrint("good")
else
setTextColor(colors.red)
sPrint("fail")
end
end
setTextColor(colors.white)
specialPrint("Unpacked to '", outputPath .. "/", "'.", colors.yellow)
end
local getSkynet = function()
if http.websocket then
-- Skynet only supports messages that are 65506 bytes or smaller
-- I'm just going with 65200 bytes to play safe.
local defineBigOnes = function(skynet)
local div = 65200
return function(channel, _message) -- big send
local message = textutils.serialize(_message)
for i = 1, math.ceil(#message / div) do
skynet.send(progdor.channel, {
msg = message:sub( (i - 1) * div + 1, i * div ),
complete = i == math.ceil(#message / div),
part = i
})
sleep(0.1)
cWrite(".", colors.lightGray)
end
end, function(channel) -- big receive
local ch, msg
local output = {}
local gotFile = false
while true do
ch, msg = skynet.receive(channel)
if type(msg) == "table" then
if type(msg.complete) == "boolean" and type(msg.msg) == "string" and type(msg.part) == "number" then
output[msg.part] = msg.msg
cWrite(".", colors.lightGray)
if msg.complete then
break
end
end
end
end
return channel, textutils.unserialize(table.concat(output))
end
end
if skynet then
local bS, bR = defineBigOnes(skynet)
skynet.open(progdor.channel)
return skynet, "", bS, bR
else
if fs.exists(progdor.skynetPath) then
local sn = dofile(progdor.skynetPath)
sn.open(progdor.channel)
local bS, bR = defineBigOnes(sn)
return sn, "", bS, bR
else
local net, contents = http.get(progdor.skynetURL)
if net then
contents = net.readAll()
local file = fs.open(progdor.skynetPath, "w")
file.write(contents)
file.close()
local sn = dofile(progdor.skynetPath)
local bS, bR = defineBigOnes(sn)
sn.open(progdor.channel)
return sn, "", bS, bR
else
return false, "Couldn't download Skynet."
end
end
end
else
return false, "This version of CC does not support Skynet."
end
end
local getModem = function()
local mod = peripheral.find("modem")
if mod then
mod.open(progdor.channel)
return mod
else
return false, "No modem was found."
end
end
local archive
local doOverwrite, doContinue = false, true
--[[ JUST SUMMIN' UP THE ELSEIF CHAIN
if mode == "api" then
elseif trMode == "transmit" then
if mode == "pack" then
end
elseif trMode == "receive" then
if mode == "pack" then
end
elseif mode == "pack" then
elseif mode == "unpack" then
elseif mode == "inspect" then
end
--]]
-- API mode takes top priority
if mode == "api" then
return {
parseArchive = parseArchive,
parseArchiveData = parseArchiveData,
buildArchive = buildArchive,
uploadToPastebin = uploadToPastebin,
}
-- after that, trans
elseif trMode == "transmit" then
-- assemble something to send
local output = {name = fs.getName(inputPath)}
if mode == "pack" then
output.contents = textutils.serialize(buildArchive(inputPath, mainFile, useCompression))
else
local file = fs.open(inputPath, "r")
output.contents = file.readAll()
file.close()
end
local grr
if useSkynet then
if not skynet then
cWrite("Connecting to Skynet...", colors.lightGray)
skynet, grr, skynetBigSend, skynetBigReceive = getSkynet()
if not skynet then
print(grr)
print("Aborting.")
return false
else
cPrint("good", colors.green)
end
end
cWrite("Sending file...", colors.lightGray)
skynetBigSend(progdor.channel, output)
skynet.socket.close()
cPrint("good", colors.green)
sWrite("Sent '")
cWrite(fs.getName(inputPath), colors.yellow)
sPrint("' using Skynet.")
else
modem, grr = getModem()
if not modem then
print(grr)
print("Abort.")
return false
end
cWrite("Sending file...", colors.lightGray)
modem.transmit(progdor.channel, progdor.channel, output)
cPrint("good", colors.green)
sWrite("Sent '")
cWrite(fs.getName(inputPath), colors.yellow)
sPrint("' using modem.")
end
elseif trMode == "receive" then
local grr
local gotFile = false
local input, channel
local didAbort = false
if useSkynet then
if not skynet then
cWrite("Connecting to Skynet...", colors.lightGray)
skynet, grr, skynetBigSend, skynetBigReceive = getSkynet()
if not skynet then
print(grr)
print("Aborting.")
return false
else
cPrint("good", colors.green)
end
end
cWrite("Waiting for file on Skynet...", colors.lightGray)
local result, grr = pcall(function()
sleep(0.05)
while true do
if parallel.waitForAny(function()
channel, input = skynetBigReceive(progdor.channel)
end, function()
local evt
while true do
evt = {os.pullEvent()}
if evt[1] == "key" then
if evt[2] == keys.q then
return
end
end
end
end) == 2 then
print("\nAbort.")
sleep(0.05)
didAbort = true
break
end
if channel == progdor.channel and type(input) == "table" then
if type(input.contents) == "string" and type(input.name) == "string" then
gotFile = true
break
end
end
end
end)
skynet.socket.close()
if not result then
error(grr, 0)
end
else
modem, grr = getModem()
if not modem then
print(grr)
print("Abort.")
sleep(0.05)
return false
end
modem.open(progdor.channel)
local evt
cWrite("Waiting for file...", colors.lightGray)
sleep(0.05)
while true do
evt = {os.pullEvent()}
if evt[1] == "modem_message" then
if evt[3] == progdor.channel and type(evt[5]) == "table" then
if type(evt[5].contents) == "string" and type(evt[5].name) == "string" then
input = evt[5]
gotFile = true
break
end
end
elseif evt[1] == "key" then
if evt[2] == keys.q then
print("\nAbort.")
sleep(0.05)
didAbort = true
break
end
end
end
end
if gotFile then
cPrint("good", colors.green)
if input.contents then
local writePath, c = fs.combine(shell.dir(), outputPath or input.name)
write("Received '")
cWrite(input.name or outputPath, colors.yellow, true)
print("'.")
if (not justOverwrite and fs.exists(writePath)) or fs.isReadOnly(writePath) then
write("\nBut, '")
cWrite(fs.getName(writePath), colors.yellow, true)
print("' is already there.")
local roCount = 0
local showROmessage = function(roCount)
if roCount == 1 then
write("\nThat file/folder is ")
cWrite("read-only", colors.yellow, true)
print("!")
elseif roCount == 2 then
write("\nI told you, that file/folder is ")
cWrite("read-only", colors.yellow, true)
print("!")
elseif roCount == 3 then
write("\nNope. The file/folder is ")
cWrite("read-only", colors.yellow, true)
print(".")
elseif roCount == 4 then
write("\nDoes the phrase ")
cWrite("read-only", colors.yellow, true)
print(" mean nothing to you?")
elseif roCount == 5 then
print("\nAlright wise-ass, that's enough.")
elseif roCount > 5 then
write("\nThat's ")
cWrite("read-only", colors.yellow, true)
print(", damn you!")
end
end
while true do
sleep(0.05)
if roCount < 5 then
write("Overwrite [Y/N]? Or [R]ename?\n")
c = choice("nry", false)
else
write("Overwrite [ /N]? Or [R]ename?\n")
c = choice("nr", false)
end
if c == 3 then
if fs.isReadOnly(writePath) then
roCount = roCount + 1
showROmessage(roCount)
else
break
end
elseif c == 1 then
print("Abort.")
return false
elseif c == 2 then
print("New name:")
if shell.dir() == "" then
write("/")
else
write("/" .. shell.dir() .. "/")
end
writePath = fs.combine(shell.dir(), read())
roCount = roCount + 1
if fs.isReadOnly(writePath) then
showROmessage(roCount)
else
break
end
end
end
end
local file = fs.open(writePath, "w")
file.write(input.contents)
file.close()
sWrite("Wrote to '")
cWrite(writePath, colors.yellow)
sPrint("'")
end
elseif not didAbort then
print("fail!")
end
elseif mode == "pack" then
if not pastebinUpload then
if fs.isReadOnly(outputPath) then
error("Output path is read-only.")
elseif fs.exists(outputPath) and (outputPath ~= inputPath) then
doContinue, doOverwrite = overwriteOutputPath(inputPath, outputPath, false, justOverwrite)
elseif fs.combine("", outputPath) == "" then
error("Output path cannot be root.")
end
if not doContinue then
return false
elseif outputPath == inputPath then
doOverwrite = true
end
end
archive = buildArchive(inputPath, mainFile, useCompression)
if exists then
if useCompression then
sPrint("Using CCA compression.")
elseif selfExtractor then
sPrint("Tacking on self-extractor.")
archive = ([[
local tArg = {...}
local selfDelete = false -- if true, deletes extractor after running
local file
local outputPath = ]] ..
(selfExtractorFolder and (
"shell.resolve(\"" .. selfExtractorFolder .. "\")"
) or (
"tArg[1] and shell.resolve(tArg[1]) or ]]" .. ((defaultAutoExtractPath and ("\"" .. defaultAutoExtractPath .. "\"")) or "shell.getRunningProgram()")
)) .. [[
local safeColorList = {[colors.white] = true,[colors.lightGray] = true,[colors.gray] = true,[colors.black] = true}
local stc = function(color) if (term.isColor() or safeColorList[color]) then term.setTextColor(color) end end
local choice = function()
local input = "yn"
write("[")
for a = 1, #input do
write(input:sub(a,a):upper())
if a < #input then
write(",")
end
end
print("]?")
local evt,char
repeat
evt,char = os.pullEvent("char")
until string.find(input:lower(),char:lower())
if verbose then
print(char:upper())
end
local pos = string.find(input:lower(), char:lower())
return pos, char:lower()
end
local archive = textutils.unserialize(]] ..
textutils.serialize(archive) ..
[[)
if fs.isReadOnly(outputPath) then
error("Output path is read-only. Abort.")
elseif fs.getFreeSpace(outputPath) <= #archive then
error("Insufficient space. Abort.")
end
]] .. ( justOverwrite and [[
if fs.exists(outputPath) and fs.combine("", outputPath) ~= "" then
fs.delete(outputPath)
end
]] or [[
if fs.exists(outputPath) and fs.combine("", outputPath) ~= "" then
print("File/folder already exists! Overwrite?")
stc(colors.lightGray)
print("(Use -o when making the extractor to always overwrite.)")
stc(colors.white)
if choice() ~= 1 then
error("Chose not to overwrite. Abort.")
else
fs.delete(outputPath)
end
end
]]
) ..
[[
if selfDelete or (fs.combine("", outputPath) == shell.getRunningProgram()) then
fs.delete(shell.getRunningProgram())
end
for name, contents in pairs(archive.data) do
stc(colors.lightGray)
write("'" .. name .. "'...")
if contents == true then -- indicates empty directory
fs.makeDir(fs.combine(outputPath, name))
else
file = fs.open(fs.combine(outputPath, name), "w")
if file then
file.write(contents)
file.close()
end
end
if file then
stc(colors.green)
print("good")
else
stc(colors.red)
print("fail")
end
end
stc(colors.white)
write("Unpacked to '")
stc(colors.yellow)
write(outputPath .. "/")
stc(colors.white)
print("'.")
]])
end
if pastebinUpload then
sWrite("Uploading to Pastebin...")
local id, url = uploadToPastebin(archive, fs.getName(inputPath))
if id then
setTextColor(colors.green)
sPrint("success!")
setTextColor(colors.white)
sPrint("Uploaded to '" .. url .. "'.")
specialPrint("Retrieve with \"", "progdor -pb " .. id .. " " .. fs.getName(inputPath), "\".", colors.yellow)
sPrint("You may need to do a Captcha on the website.")
if not fs.exists(progdor.PBlogPath) then
setTextColor(colors.lightGray)
specialPrint("(PB uploads are logged at \"", progdor.PBlogPath, "\".)", colors.yellow)
setTextColor(colors.white)
end
-- precautionary log file
local file = fs.open(progdor.PBlogPath, "a")
file.writeLine("uploaded \"" .. inputPath .. "\" to \"" .. url .. "\"")
file.close()
else
sPrint("failed!")
end
else
if doOverwrite then
fs.delete(outputPath)
end
local file = fs.open(outputPath, "w")
file.write(archive)
file.close()
if selfExtract then
specialPrint("Written self-extractor to '", outputPath, "'.", colors.yellow)
else
specialPrint("Written to '", outputPath, "'.", colors.yellow)
end
end
else
error("No such input path exists.")
return false
end
elseif mode == "unpack" then -- unpack OR upload
if pastebinUpload then
local file = fs.open(inputPath, "r")
archive = file.readAll()
file.close()
sWrite("Uploading to Pastebin...")
local id, url = uploadToPastebin(archive, fs.getName(inputPath))
if id then
setTextColor(colors.green)
sPrint("success!")
setTextColor(colors.white)
sPrint("Uploaded to '" .. url .. "'.")
specialPrint("Retrieve with \"", "progdor -pb " .. id .. " " .. fs.getName(inputPath), "\".", colors.yellow)
sPrint("You may need to do a Captcha on the website.")
if not fs.exists(progdor.PBlogPath) then
setTextColor(colors.lightGray)
specialPrint("(PB uploads are logged at \"", progdor.PBlogPath, "\".)", colors.yellow)
setTextColor(colors.white)
end
-- precautionary log file
local file = fs.open(progdor.PBlogPath, "a")
file.writeLine("uploaded \"" .. inputPath .. "\" to \"" .. url .. "\"")
file.close()
else
setTextColor(colors.red)
sPrint("failed!")
setTextColor(colors.white)
return false
end
elseif pastebinGet or directDownload then
local url, contents
if pastebinGet and directDownload then
error("Cannot do both pastebin get and direct download.")
elseif fs.isReadOnly(outputPath) then
error("Output path is read-only.")
elseif fs.combine(outputPath, "") == "" then
error("Output path cannot be root.")
else
if pastebinGet then
url = "http://www.pastebin.com/raw/" .. pastebinGet
elseif directDownload then
url = directDownload
end
if fs.exists(outputPath) and (outputPath ~= inputPath) or outputPath == shell.getRunningProgram() then
doContinue, doOverwrite = overwriteOutputPath(inputPath, outputPath, true, justOverwrite)
end
if not doContinue then
return false
elseif outputPath == inputPath then
doOverwrite = true
end
sWrite("Connecting to \"")
setTextColor(colors.yellow)
sWrite(url)
setTextColor(colors.white)
sWrite("\"...")
local handle = http.get(url)
if handle then
cPrint("success!", colors.green)
contents = handle.readAll()
handle.close()
-- detects if you didn't solve the captcha, since archives commonly trigger anti-spam measures
if (
pastebinGet and
(not textutils.unserialize(contents)) and
contents:find("Your paste has triggered our automatic SPAM detection filter.")
) then
specialPrint("You must go to '", url, "' and do the Captcha to be able to download that paste.", colors.yellow)
return false
end
setTextColor(colors.lightGray)
sWrite("Parsing archive...")
archive = parseArchiveData(contents)
if archive then
setTextColor(colors.green)
sPrint("good")
else
setTextColor(colors.red)
sPrint("Invalid archive file.")
return false
end
if doOverwrite then
fs.delete(outputPath)
end
writeArchiveData(archive, outputPath)
else
setTextColor(colors.red)
sPrint("failed!")
setTextColor(colors.white)
return false
end
end
else -- regular unpack
if exists then
if fs.isReadOnly(outputPath) then
error("Output path is read-only.")
elseif fs.exists(outputPath) and (outputPath ~= inputPath) or outputPath == shell.getRunningProgram() then
doContinue, doOverwrite = overwriteOutputPath(inputPath, outputPath, true, justOverwrite)
end
if not doContinue then
return false
elseif outputPath == inputPath then
doOverwrite = true
end
setTextColor(colors.lightGray)
sWrite("Parsing archive...")
archive = parseArchive(inputPath)
if archive then
setTextColor(colors.green)
sPrint("good")
if doOverwrite then
fs.delete(outputPath)
end
writeArchiveData(archive, outputPath)
else
setTextColor(colors.red)
sPrint("Invalid archive file.")
return false
end
else
error("No such input path exists.")
end
end
elseif mode == "inspect" then
if exists and (not fs.isDir(inputPath)) then
archive = parseArchive(inputPath, true)
local totalSize = 0
local amountOfFiles = 0
local averageSize = 0
local output = {}
if archive then
for k,v in pairs(archive) do
if k == "data" then
for name, contents in pairs(v) do
if contents then -- don't count directories, where contents == false
totalSize = totalSize + #contents
amountOfFiles = amountOfFiles + 1
end
end
averageSize = math.ceil(totalSize / amountOfFiles)
else
output[#output + 1] = k .. " = \"" .. tostring(v) .. "\""
end
end
sPrint("# of files: " .. amountOfFiles)
sPrint("Total size: " .. totalSize .. " bytes (" .. round(totalSize / 1024, 1) .. " KB)")
sPrint("Aveg. size: " .. averageSize .. " bytes (" .. round(averageSize / 1024, 1) .. " KB)")
sPrint(("-"):rep(scr_x))
for i = 1, #output do
sPrint(output[i])
end
else
error("Invalid archive file.")
end
else
if fs.isDir(inputPath) then
error("Cannot inspect directories.")
else
error("No such input path exists.")
end
end
end