Advertisement
MindenCucc

MakePack

Jun 7th, 2015
83
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Lua 16.03 KB | None | 0 0
  1. --[[
  2.  
  3. MakePack by MarcusD
  4. http://www.computercraft.info/forums2/index.php?/topic/23513-
  5.  
  6. --]]
  7.  
  8. local argz = {...}
  9.  
  10. --[HARDCODED VARIABLES]--
  11.  
  12. local isapi = shell == nil
  13.  
  14. --lineComp
  15. local lineCompOffs = 0
  16. local lineCompNum = 9
  17.  
  18. --lineExt
  19. local lineExtOffs = 0
  20. local lineExtOffsEnd = -0
  21.  
  22. --[HARDCODED VARIABLES]--
  23.  
  24. if #argz == 0 or argz[1] == "help" and not isapi then
  25.   do
  26.     local str =
  27.     [[
  28. MakePack by MarcusD
  29.  
  30. Usage:
  31.  
  32. makepack help -- Displays this message
  33. makepack pack folder to_file -- Packs folder into to_file
  34. makepack unpack file to_folder -- Unpacks file to to_folder
  35. makepack sfx folder to_file -- Creates a SelF-eXtracting archive
  36.  
  37. Note:
  38.  
  39. Please don't use this on recursive symlinks, or else... >.<
  40.    ]]
  41.    print(str)
  42.    return nil
  43.  end
  44. end
  45.  
  46. local function base_convert(n, b)
  47.  if type(n) == "string" then return tonumber(n, b) end
  48.  n = math.floor(n)
  49.  if not b or b == 10 then return tostring(n) end
  50.  local digits = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ"
  51.  local t = {}
  52.  if n < 0 then
  53.    table.insert(t, "-")
  54.    n = -n
  55.  end
  56.  repeat
  57.    local d = (n % b) + 1
  58.    n = floor(n / b)
  59.    insert(t, 1, digits:sub(d,d))
  60.  until n == 0
  61.  return table.concat(t,"")
  62. end
  63.  
  64. local function iter_w(whand, from, comp)
  65.  local function wint(x)
  66.    local b4=x%256 x=(x-x%256)/256
  67.    local b3=x%256 x=(x-x%256)/256
  68.    local b2=x%256 x=(x-x%256)/256
  69.    local b1=x%256 x=(x-x%256)/256
  70.    whand.write(b1)
  71.    whand.write(b2)
  72.    whand.write(b3)
  73.    whand.write(b4)
  74.  end
  75.  
  76.  local function wstr(str)
  77.    whand.write(string.len(str))
  78.    for i in str:gmatch(".") do
  79.      whand.write(string.byte(i))
  80.    end
  81.  end
  82.  
  83.  local buf = {}
  84.  local strbuf = ""
  85.  
  86.  if fs.exists(from) and not fs.isDir(from) then
  87.    wstr(fs.getName(from)) -- name
  88.    whand.write(1) -- isfile
  89.    local rhand = fs.open(from, "rb")
  90.    local nao = rhand.read()
  91.    while nao do --because there's no rhand.size and fs.size is inaccurate
  92.       table.insert(buf, string.char(nao))
  93.       nao = rhand.read()
  94.     end
  95.     rhand.close()
  96.     os.queueEvent("keepalive")
  97.     os.pullEvent("keepalive")
  98.     strbuf = comp.encode(table.concat(buf, ""))
  99.     os.queueEvent("keepalive")
  100.     os.pullEvent("keepalive")
  101.     wint(string.len(strbuf)) -- file size
  102.     for asd in strbuf:gmatch(".") do
  103.       whand.write(string.byte(asd))
  104.     end
  105.     os.queueEvent("keepalive")
  106.     os.pullEvent("keepalive")
  107.     buf = {}
  108.     strbuf = ""
  109.     whand.flush()
  110.     return nil
  111.   end
  112.  
  113.   local function iter(pth, short)
  114.     os.queueEvent("keepalive")
  115.     os.pullEvent("keepalive")
  116.     if not short then short = "/" end
  117.     local tbl = fs.list(pth)
  118.     local cnt = #tbl
  119.     wstr(short) -- name
  120.     whand.write(0) -- isfile
  121.     wint(cnt) -- #entries / if not isfile
  122.     for _, v in pairs(tbl) do
  123.       local pat = fs.combine(pth, v)
  124.       if not isapi then print("Processing " .. pat) end
  125.       if fs.isDir(pat) then
  126.         iter(pat, v)
  127.       else
  128.         wstr(v) -- name
  129.         whand.write(1) -- isfile
  130.         local rhand = fs.open(pat, "rb")
  131.         local nao = rhand.read()
  132.         while nao do --because there's no rhand.size and fs.size is inaccurate
  133.           table.insert(buf, string.char(nao))
  134.           nao = rhand.read()
  135.         end
  136.         rhand.close()
  137.         os.queueEvent("keepalive")
  138.         os.pullEvent("keepalive")
  139.         strbuf = comp.encode(table.concat(buf, ""))
  140.         os.queueEvent("keepalive")
  141.         os.pullEvent("keepalive")
  142.         wint(string.len(strbuf)) -- file size
  143.         for asd in strbuf:gmatch(".") do
  144.           whand.write(string.byte(asd))
  145.         end
  146.         os.queueEvent("keepalive")
  147.         os.pullEvent("keepalive")
  148.         buf = {}
  149.         strbuf = ""
  150.         whand.flush()
  151.       end
  152.     end
  153.   end
  154.  
  155.   return iter(from)
  156. end
  157.  
  158. local function errnum(str)
  159.   return tonumber(string.match(str, ":([0-9%.]+):"))
  160. end
  161.  
  162. local function lineComp() error("thing") end
  163. local comp =[[
  164. --Original location: http://www.wowace.com/addons/libcompress/
  165. --Heavily modified and minified
  166. local type=type;local select=select;local next=next;local loadstring=loadstring;local setmetatable=setmetatable;local assert=assert;local a=table.insert;local b=table.remove;local c=table.concat;local d=table.sort;local e=string.char;local f=string.byte;local g=string.len;local h=string.sub;local unpack=unpack;local pairs=pairs;local i=math.modf;local j=bit.band;local k=bit.bor;local l=bit.bxor;local m=bit.bnot;local n=bit.blshift;local o=bit.brshift;local p={}local function q(r)for s=1,#p do p[s]=nil end;local t;r,t=i(r/255)t=t*255;p[#p+1]=t;while r>0 do r,t=i(r/255)t=t*255;p[#p+1]=t end;if#p==1 and p[1]>0 and p[1]<250 then return e(p[1])else for u=1,#p do p[u]=p[u]+1 end;return e(256-#p,unpack(p))end end;local function v(w,u)u=u or 1;local x=f(w,u,u)if x>249 then local y=0;x=256-x;for z=u+x,u+1,-1 do y=y*255+f(w,z,z)-1 end;return y,x+1 else return x,1 end end;local A={}local function B(C)if type(C)=="string"then local D=256;A={}local E={"\002"}local F=''local G=1;for u=0,255 do A[e(u)]=u end;for u=1,#C do local H=C:sub(u,u)local I=F..H;if A[I]then F=I else A[I]=D;D=D+1;local y=q(A[F])G=G+#y;E[#E+1]=y;F=H end end;if F then local y=q(A[F])G=G+#y;E[#E+1]=y end;if#C+1>G then return c(E)else return e(1)..C end else return nil,"Can only compress strings"end end;local function J(K)if type(K)=="string"then if K:sub(1,1)~="\002"then return nil,"Can only decompress LZW compressed data ("..tostring(K:sub(1,1))..")"end;K=K:sub(2)local D=256;for s in pairs(A)do A[s]=nil end;for u=0,255 do A[u]=e(u)end;local E={}local L=1;local M,s;s,M=v(K,L)L=L+M;E[#E+1]=A[s]local F=A[s]local N;while L<=#K do s,M=v(K,L)L=L+M;N=A[s]or F..F:sub(1,1)E[#E+1]=N;A[D]=F..N:sub(1,1)D=D+1;F=N end;return c(E)else return nil,"Can only uncompress strings"end end;local function O(P,Q,R)if P then P.bcode=Q;P.blength=R;if P.c1 then O(P.c1,k(Q,n(1,R)),R+1)end;if P.c2 then O(P.c2,Q,R+1)end end end;local function S(T,R)local U=0;local V;local W=0;for u=R-1,0,-1 do V=j(T,n(1,u))==0 and 0 or 1;U=n(U,1+V)+V;W=W+V end;return U,R+W end;local X=0;local Y;local Z;local function _(a0,T,R)Y=Y+n(T,Z)Z=R+Z;if Z>32 then return true end;while Z>=8 do X=X+1;a0[X]=e(j(Y,255))Y=o(Y,8)Z=Z-8 end end;local function a1(C)if type(C)~="string"then return nil,"Can only compress strings"end;if#C==0 then return"\001"end;local a2={}local z=0;local a3=g(C)local H;for u=1,a3 do H=f(C,u)a2[H]=(a2[H]or 0)+1 end;local a4={}local a5;local a6={}for a7,a8 in pairs(a2)do a5={symbol=e(a7),weight=a8}a6[a7]=a5;a(a4,a5)end;d(a4,function(x,V)if x.weight<V.weight then return true elseif x.weight>V.weight then return false else return nil end end)local a9=#a4;local aa={}local R,ab,ac,ad,ae,af;local ag;while#a4+#aa>1 do if not next(aa)then ac,ae=next(a4)b(a4,ac)elseif not next(a4)then ad,ae=next(aa)b(aa,ad)else ac,R=next(a4)ad,ab=next(aa)if R.weight<=ab.weight then ae=R;b(a4,ac)else ae=ab;b(aa,ad)end end;if not next(aa)then ac,af=next(a4)b(a4,ac)elseif not next(a4)then ad,af=next(aa)b(aa,ad)else ac,R=next(a4)ad,ab=next(aa)if R.weight<=ab.weight then af=R;b(a4,ac)else af=ab;b(aa,ad)end end;ag={c1=ae,c2=af,weight=ae.weight+af.weight}a(aa,ag)end;if#a4>0 then ac,R=next(a4)a(aa,R)b(a4,ac)end;aa=aa[1]O(aa,0,0)if aa then aa.bcode=0;aa.blength=1 end;Y=0;Z=0;local K={}K[1]="\003"local R=g(C)K[2]=e(j(a9-1,255))K[3]=e(j(R,255))K[4]=e(j(o(R,8),255))K[5]=e(j(o(R,16),255))X=5;for a7,a5 in pairs(a6)do _(K,a7,8)if _(K,S(a5.bcode,a5.blength))then return e(0)..C end;_(K,3,2)end;local ah={}local ai=0;local aj;for u=1,R,200 do aj=R<u+199 and R or u+199;for ak=u,aj do H=f(C,ak)_(K,a6[H].bcode,a6[H].blength)end;ai=ai+1;ah[ai]=c(K,"",1,X)X=0 end;if Z>0 then ai=ai+1;ah[ai]=e(Y)end;local al=c(ah,"",1,ai)if#C+1<=#al then return"\001"..C end;return al end;local am={}setmetatable(am,{__index=function(L,s)local an=n(1,s)rawset(L,s,an)return an end})local ao={}setmetatable(ao,{__index=function(L,s)local an=n(1,s)-1;rawset(L,s,an)return an end})local function ap(aq,ar)if ar>=2 then local V;local as=0;for u=0,ar-1 do V=j(aq,am[u])if not(as==0)and not(V==0)then return j(aq,ao[u-1]),u-1,o(aq,u+1),ar-u-1 end;as=V end end;return nil end;local function at(T,au)local av=0;local V;local W=0;local u=0;while u<au do V=j(T,am[u])if not(V==0)then av=k(av,am[W])u=u+1 end;u=u+1;W=W+1 end;return av,W end;local function aw(K)if not type(C)=="string"then return nil,"Can only uncompress strings"end;local X=#K;local ax=f(K)if ax==1 then return K:sub(2)end;if not(ax==3)then return nil,"Can only decompress Huffman compressed data ("..tostring(ax)..")"end;local ay=f(h(K,2,2))+1;local az=f(h(K,3,3))local c1=f(h(K,4,4))local c2=f(h(K,5,5))local aA=c2*65536+c1*256+az;if aA==0 then return""end;local aq=0;local aB=0;local aC={}setmetatable(aC,{__index=function(L,s)local an={}rawset(L,s,an)return an end})local u=6;local H,aD;local aE=1000;local aF=0;local a7,T,au,aG,aH;local z=0;local aI=0;while z<ay do if u>X then return nil,"Cannot decode map"end;H=f(K,u)aq=k(aq,n(H,aB))aB=aB+8;if aI==0 then a7=j(aq,255)aq=o(aq,8)aB=aB-8;aI=1 else T,au,aG,aH=ap(aq,aB)if T then aq,aB=aG,aH;H,aD=at(T,au)aC[aD][H]=e(a7)aE=aD<aE and aD or aE;aF=aD>aF and aD or aF;z=z+1;aI=0 end end;u=u+1 end;local aJ={}setmetatable(aC,{__index=function(L,s)return aJ end})local C={}local aK={}local a3=0;local aL=0;local aM;local aN=aE;local a7;local aO=0;X=X+1;local aP=200;aP=aP>aA and aA or aP;while true do if aN<=aB then aM=j(aq,ao[aN])a7=aC[aN][aM]if a7 then a3=a3+1;C[a3]=a7;aO=aO+1;if aO>=aP then if aO>=aA then break end;aL=aL+1;aK[aL]=c(C,"",1,a3)a3=0;aP=aP+200;aP=aP>aA and aA or aP end;aq=o(aq,aN)aB=aB-aN;aN=aE else aN=aN+1;if aN>aF then return nil,"Decompression error at "..tostring(u).."/"..tostring(#K)end end else H=f(K,u)aq=aq+n(H or 0,aB)aB=aB+8;if u>X then break end;u=u+1 end end;local aQ={}a(aQ,c(aK,"",1,aL))a(aQ,c(C,"",1,a3))return c(aQ,"")end;local function aR(aS)if type(aS)~="string"then return nil,"Can only handle strings"end;if string.byte(aS)~=1 then return nil,"Can only handle uncompressed data"end;return aS:sub(2)end;local aT={[2]=B,[3]=a1}local aU={[1]=aR,[2]=J,[3]=aw}local function aV(aS)local aW=next(aT)local E=aT[aW](aS)local z;aW=next(aT,aW)while aW do z=aT[aW](aS)if#z<#E then E=z end;aW=next(aT,aW)end;return E end;local function aX(aS)local aY=string.byte(aS)if aU[aY]then return aU[aY](aS)else return nil,"Unknown compression method ("..tostring(aY)..")"end end;return{lzw_c=B,lzw_d=J,huff_c=a1,huff_d=aw,encode=aV,decode=aX}
  167. ]]
  168. local ejj
  169. comp, ejj = loadstring(comp)
  170. if not comp then error(ejj) end
  171. comp = comp()
  172.  
  173. local choice = argz[1]
  174. if choice == "pack" then
  175.   if #argz < 3 then error("Invalid number of parameters supplied: " .. #argz) end
  176.   local from = argz[2]
  177.   local to = argz[3]
  178.   assert(fs.exists(from), "Source does not exists")
  179.   --assert(fs.isDir(from), "Only directories are supported now")
  180.   assert(not fs.exists(to), "Destination exists")
  181.  
  182.   local whand = fs.open(to, "wb")
  183.  
  184.   iter_w(whand, from, comp)
  185.  
  186.   whand.close()
  187.  
  188.   if not isapi then print("Successfully created archive!") end
  189.   return true
  190. end
  191. local function lineExt() error("thing") end
  192. local function iter_r(rhand, from, comp)
  193.   local function rint()
  194.     local b1, b2, b3, b4 = rhand.read(), rhand.read(), rhand.read(), rhand.read()
  195.     return b1*16777216 + b2*65536 + b3*256 + b4
  196.   end
  197.  
  198.   local function rstr()
  199.     local buf = {}
  200.     local cnt = rhand.read()
  201.     for i = 1, cnt do
  202.       table.insert(buf, string.char(rhand.read()))
  203.     end
  204.     return table.concat(buf, "")
  205.   end
  206.  
  207.   local buf = {}
  208.  
  209.   local function iter(root, pth)
  210.     os.queueEvent("keepalive")
  211.     os.pullEvent("keepalive")
  212.     local fpat = fs.combine(root, pth)
  213.     local name = rstr()
  214.     local nid = rhand.read()
  215.     if not isapi then print("Extracting " .. (nid == 0 and "dir" or "dat") .. " " .. fs.combine(pth, name)) end
  216.     if nid == 0 then -- dir
  217.       local cnt = rint()
  218.       fs.makeDir(fs.combine(fpat, name))
  219.       local relpth = fs.combine(pth, name)
  220.       for i = 1, cnt do
  221.         iter(root, relpth)
  222.       end
  223.     elseif nid == 1 then -- file
  224.       local len = rint()
  225.       local whand = fs.open(fs.combine(fpat, name), "wb")
  226.       for i = 1, len do
  227.         if i % 256 == 0 then
  228.           os.queueEvent("keepalive")
  229.           os.pullEvent("keepalive")
  230.         end
  231.         table.insert(buf, string.char(rhand.read()))
  232.       end
  233.       os.queueEvent("keepalive")
  234.       os.pullEvent("keepalive")
  235.       for chr in comp.decode(table.concat(buf, "")):gmatch(".") do
  236.         whand.write(string.byte(chr))
  237.       end
  238.       whand.flush()
  239.       whand.close()
  240.       buf = {}
  241.     else
  242.       error("Invalid node type: " .. nid)
  243.     end
  244.   end
  245.  
  246.   return iter(from, "/")
  247. end
  248. if choice == "unpack" then
  249.   if #argz < 3 then error(argz[4] and "No destination given" or ("Invalid number of parameters supplied: " .. #argz)) end
  250.   local from = argz[2]
  251.   local to = argz[3]
  252.   assert(fs.exists(from), "Source does not exists")
  253.   assert(not fs.isDir(from), "Not a valid makeself file")
  254.   assert(not fs.exists(to), "Destination exists")
  255.   local rhand = fs.open(from, "rb")
  256.   if argz[4] == true then while rhand.read() ~= 0 do end end
  257.   iter_r(rhand, to, comp)
  258.   rhand.close()
  259.   if not isapi then print("Successfully extracted archive!") end
  260.   return true
  261. end
  262. local function lineExtEnd() error("thing") end
  263. if choice == "sfx" then
  264.   if #argz < 3 then error("Invalid number of parameters supplied: " .. #argz) end
  265.   local from = argz[2]
  266.   local to = argz[3]
  267.   assert(fs.exists(from), "Source does not exists")
  268.   --assert(fs.isDir(from), "Only directories are supported now")
  269.   assert(not fs.exists(to), "Destination exists")
  270.  
  271.   local disfeil = ""
  272.   if shell then
  273.     disfeil = shell.getRunningProgram()
  274.   else
  275.     if argz[4] then
  276.       disfeil = argz[4]
  277.     else
  278.       print("Please supply the location to the makepack executable")
  279.       disfeil = read()
  280.     end
  281.   end
  282.   assert(not fs.isDir(disfeil), "The supplied file is a directory")
  283.   assert(fs.exists(disfeil), "File not exists: " .. disfeil)
  284.  
  285.   local rhand = fs.open(disfeil, "r")
  286.   local whand = fs.open(to, "w")
  287.  
  288.   whand.writeLine("local argz = {...} argz[3] = argz[1] argz[2] = shell.getRunningProgram() argz[4] = true local choice=\"unpack\"")
  289.  
  290.   local num1, num2, num3, num2Diff, num3Diff
  291.  
  292.   local asd, errloc = pcall(lineComp)
  293.   num1 = errnum(errloc)
  294.   _, errloc = pcall(lineExt)
  295.   num2 = errnum(errloc)
  296.   _, errloc = pcall(lineExtEnd)
  297.   num3 = errnum(errloc)
  298.  
  299.   for i = 1, num1 do
  300.     rhand.readLine()
  301.   end
  302.  
  303.   for i = 1, lineCompOffs do
  304.     rhand.readLine()
  305.   end
  306.  
  307.   for i = 1, lineCompNum do
  308.     whand.writeLine(rhand.readLine())
  309.   end
  310.  
  311.   num2Diff = num2 - (num1 + lineCompOffs + lineCompNum)
  312.  
  313.   for i = 1, num2Diff do
  314.     rhand.readLine()
  315.   end
  316.  
  317.   for i = 1, lineExtOffs do
  318.     rhand.readLine()
  319.   end
  320.  
  321.   num3Diff = num3 - lineExtOffs + lineExtOffsEnd - num2 - 1
  322.  
  323.   for i = 1, num3Diff do
  324.     whand.writeLine(rhand.readLine())
  325.   end
  326.   whand.write("--[==[\000")
  327.  
  328.   whand.flush()
  329.   rhand.close()
  330.   whand.close()
  331.  
  332.   whand = fs.open(to, "ab")
  333.  
  334.   iter_w(whand, from, comp)
  335.  
  336.   whand.write(string.byte("]"))
  337.   whand.write(string.byte("="))
  338.   whand.write(string.byte("="))
  339.   whand.write(string.byte("]"))
  340.  
  341.   whand.flush()
  342.   whand.close()
  343.  
  344.   print("SelF-eXtracting archive created")
  345.   return true
  346. elseif choice == "api" then
  347.   isapi = true
  348.   return
  349.   {
  350.     compress = function(wat, where)
  351.       assert(fs.exists(wat), "Source does not exists")
  352.       --assert(fs.isDir(wat), "Only directories are supported now")
  353.       assert(not fs.exists(where), "Destination exists")
  354.       local whand = fs.open(where, "wb")
  355.       iter_w(whand, wat, comp)
  356.       whand.flush()
  357.       whand.close()
  358.       return true
  359.     end,
  360.    
  361.     decompress = function(wat, where, hazsfx)
  362.       assert(fs.exists(wat), "Source does not exists")
  363.       assert(not fs.isDir(wat), "Not a valid makeself file")
  364.       assert(not fs.exists(where), "Destination exists")
  365.       local rhand = fs.open(wat, "rb")
  366.       if hazsfx then while rhand.read() ~= 0 do end end
  367.       iter_r(rhand, where, comp)
  368.       rhand.close()
  369.       return true
  370.     end
  371.   }
  372. else
  373.   error("Invalid option: " .. tostring(choice))
  374. end
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement