Guest User

Quarry 3.2.3 Ender Chest

a guest
Nov 15th, 2013
92
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
text 39.07 KB | None | 0 0
  1. --[[
  2. Version 3.2.3
  3. Recent Changes:
  4. 1. Now has session persistence! Just type "quarry -restore"
  5. and your turtle will pick up where it left off
  6. 2. New Argument: startY to help stop it from getting stuck on those nasty sheep!
  7. 3. Most directional functions take care of invert. Exception is turtle.detect
  8. ]]
  9. --[[ToDo:
  10. 1. Actually get the rednet send back and forth to work
  11. 3. Send basic commands from receiver program e.g. Stop
  12. 6. Add in rednet stuff
  13. 7. Massive update for rednet program
  14. ]]
  15. --Defining things
  16. _G.civilTable = {}; civilTable = nil; setmetatable(civilTable, {__index = _G}); setfenv(1,civilTable)
  17. -------Defaults for Arguments----------
  18. --Arguments assignable by text
  19. x,y,z = 3,3,3 --These are just in case tonumber fails
  20. inverted = false --False goes from top down, true goes from bottom up [Default false]
  21. rednetEnabled = false --Default rednet on or off [Default false]
  22. --Arguments assignable by tArgs
  23. dropSide = "front" --Side it will eject to when full or done [Default "front"]
  24. careAboutResources = true --Will not stop mining once inventory full if false [Default true]
  25. doCheckFuel = true --Perform fuel check [Default true]
  26. doRefuel = false --Whenever it comes to start location will attempt to refuel from inventory [Default false]
  27. invCheckFreq = 10 --Will check for inventory full every <-- moved spaces [Default 10]
  28. keepOpen = 1 --How many inventory slots it will attempt to keep open at all times [Default 1]
  29. fuelSafety = "moderate" --How much fuel it will ask for: safe, moderate, and loose [Default moderate]
  30. saveFile = "Civil_Quarry_Restore" --Where it saves restore data [Default "Civil_Quarry_Restore"]
  31. doBackup = true --If it will keep backups for session persistence [Default true]
  32. numberOfStacksPerRun = 8 --How many stacks (number of items) the turtle expects (on average) to have before it must dump off. Not in arguments. [Default 8]
  33. gpsEnabled = false -- If option is enabled, will attempt to find position via GPS api [Default false]
  34. gpsTimeout = 3 --The number of seconds the program will wait to get GPS coords. Not in arguments [Default 3]
  35. logging = true --Whether or not the turtle will log mining runs. [Default ...still deciding]
  36. logFolder = "Quarry_Logs" --What folder the turtle will store logs in [Default "Quarry_Logs"]
  37. logExtension = "" --The extension of the file (e.g. ".txt") [Default ""]
  38. --Standard number slots for fuel (you shouldn't care)
  39. fuelTable = { --Will add in this amount of fuel to requirement.
  40. safe = 1000,
  41. moderate = 200,
  42. loose = 0 } --Default 1000, 200, 0
  43. --Standard rednet channels
  44. channels = {
  45. send = os.getComputerID() ,
  46. receive = os.getComputerID() + 100 ,
  47. confirm = "Confirm"
  48. }
  49.  
  50. local help_paragraph = [[
  51. -DEFAULT: This will ignore all arguments and prompts and use defaults
  52. -vanilla: This will ignore all arguments except for dimensions
  53. -dim: [num] [num] [num] This sets the dimensions of the quarry
  54. -invert: [t/f] This sets whether invert is true or false
  55. -rednet: [t/f] This sets whether it will attempt to make a rednet connection
  56. -sendChannel: [num] This sets the channel the turtle will attempt to send on
  57. -receiveChannel: [num] This sets the channel the turtle will attempt to receive on
  58. -doRefuel: Changes whether or not the turtle will refuel itself with coal when fuel is low (opposite of written config)
  59. -doCheckFuel: Changes whether or not the turtle will check its fuel level before running (opposite of written config)
  60. -chest: [chest side] Changes what side turtle will output to
  61. -startY: [num] Tells the turtle what y coordinate it starts at, to make bedrock checking more accurate (hopefully)
  62. -keepOpen: [num] How many slots of the inventory turtle will try to keep open (Will then auto-empty)
  63. -invCheckFreq: [num] How many blocks before full inventory checks
  64. -saveFile: [word] Changes where the turtle saves its backup to.
  65. -doBackup: [t/f] Whether or not the turtle will backup its current position to a file.
  66. -restore: Turtle will check for a save file. If found will ignore all other arguments.
  67. -fuelSafety: [safe/moderate/loose] How much extra fuel the turtle will request at startup
  68. -GPS: Will toggle whether gps is on or not.
  69. You must have a valid GPS network set up for this to work. It will take its starting location from this and its first position.
  70. It will then store these values in its progress file in case your turtle stops for any reason.
  71. On startup, the turtle will calculate its current position based on a new gps locate.
  72. -log: Turns logging on if not already on.
  73. Will publish reports of mining runs to its own folder for later reading
  74. Smart Fueling System: As of 3.2.0, the turtle uses a new smart fueling system
  75. With this new system you can shift click in as many stacks of fuel as you want, and it will take stacks of fuel at a time
  76. This makes fueling for large jobs on new turtles much, much faster.
  77. Examples [1]:
  78. The minimum amount necessary to start a turtle automatically would be one of these two
  79. ---------
  80. quarry -dim 3 3 3 -invert false -rednet false
  81. ---------
  82. quarry -dim 3 3 3 -vanilla
  83. ---------
  84. or, if you actually wanted a 3x3x3 then
  85. ---------
  86. quarry -DEFAULT
  87. Examples [2]:
  88. If you wanted start a quarry that has
  89. rednet on channels 500 and 501, outputs
  90. to a chest below itself, is inverted,
  91. and has the dimesions 5x5x21, then here
  92. is what you would type: (Remember that
  93. order does not matter)
  94. ----------
  95. quarry -receiveChannel 501 -sendChannel 500 -invert true -dim 5 5 21 -chest bottom -rednet true
  96. Examples [2] (cont.):
  97. Now, this may be very long, but it is meant for people who want to make automated quarries with
  98. the same settings every time (or with variable setting)
  99. Examples [3]:
  100. If you are playing softcore then you
  101. can do
  102. ---------
  103. quarry -doCheckFuel false
  104. ---------
  105. Followed by any other arguments
  106. Tips:
  107. You don't actually have to type out "false" or "true" if you don't want. It actaully just checks if the first
  108. letter is "t", so you (usually) don't have to put anything at all. Putting the whole word just helps with clarity
  109. Internal Config:
  110. At the top of the program, right below the changelog is a written config. Anything not specified by arguments
  111. will be taken from here. If you have a quarry setup you use a lot, you could just edit the config and then
  112. type in the following:
  113. ----------
  114. quarry -DEFAULT
  115. ]]
  116.  
  117. --Parsing help for display
  118. --[[The way the help table works:
  119. All help indexes are numbered. There is a help[i].title that contains the title,
  120. and the other lines are in help[i][1] - help[i][#help[i] ]
  121. Different lines (e.g. other than first) start with a space.
  122. As of now, the words are not wrapped, fix that later]]
  123. local help = {}
  124. local i = 0
  125. local titlePattern = ".-%:" --Find the beginning of the line, then characters, then a ":"
  126. local textPattern = "%:.+" --Find a ":", then characters until the end of the line
  127. for a in help_paragraph:gmatch("\n?.-\n") do --Matches in between newlines
  128. local current = string.sub(a,1,-2).."" --Concatenate Trick
  129. if string.sub(current,1,1) ~= " " then
  130. i = i + 1
  131. help[i] = {}
  132. help[i].title = string.sub(string.match(current, titlePattern),1,-2)..""
  133. help[i][1] = string.sub(string.match(current,textPattern) or " ",3,-1)
  134. elseif string.sub(current,1,1) == " " then
  135. table.insert(help[i], string.sub(current,2, -1).."")
  136. end
  137. end
  138.  
  139.  
  140. local supportsRednet = (peripheral.wrap("right") ~= nil)
  141.  
  142. local tArgs = {...}
  143. --You don't care about these
  144. xPos,yPos,zPos,facing,percent,mined,moved,relxPos, rowCheck, connected, isInPath, layersDone, attacked, startY
  145. = 0, 1, 1, 0, 0, 0, 0, 1, "right", false, true, 1, 0, 0
  146.  
  147. local totals = {cobble = 0, fuel = 0, other = 0} -- Total for display (cannot go inside function)
  148. local function count() --Done any time inventory dropped and at end
  149. slot = {} --1: Cobble 2: Fuel 3:Other
  150. for i=1, 16 do --[1] is type, [2] is number
  151. slot[i] = {}
  152. slot[i][2] = turtle.getItemCount(i)
  153. end
  154. slot[1][1] = 1 -- = Assumes Cobble/Main
  155. for i=1, 16 do --Cobble Check
  156. turtle.select(i)
  157. if turtle.compareTo(1) then
  158. slot[i][1] = 1
  159. totals.cobble = totals.cobble + slot[i][2]
  160. elseif turtle.refuel(0) then
  161. slot[i][1] = 2
  162. totals.fuel = totals.fuel + slot[i][2]
  163. else
  164. slot[i][1] = 3
  165. totals.other = totals.other + slot[i][2]
  166. end
  167. end
  168. turtle.select(1)
  169. end
  170.  
  171. local getFuel = turtle.getFuelLevel --This is for cleanup at the end
  172. do --Common variable name...
  173. local flag = turtle.getFuelLevel() == "Unlimited"--Unlimited screws up my calculations
  174. if flag then --Fuel is disabled
  175. turtle.getFuelLevel = function() return math.huge end --Infinite Fuel
  176. end --There is no "else" because it will already return the regular getFuel
  177. end
  178. local checkFuel = turtle.getFuelLevel --Just an alias for backwards compat
  179.  
  180. -----------------------------------------------------------------
  181. --Input Phase
  182. local function screen(xPos,yPos)
  183. xPos, yPos = xPos or 1, yPos or 1
  184. term.setCursorPos(xPos,yPos); term.clear(); end
  185. local function screenLine(xPos,yPos)
  186. term.setCursorPos(xPos,yPos); term.clearLine(); end
  187.  
  188. screen(1,1)
  189. print("----- Welcome to Quarry! -----")
  190. print("")
  191.  
  192. local sides = {top = "top", right = "right", left = "left", bottom = "bottom", front = "front"} --Used to whitelist sides
  193. local errorT = {num = "Numbers not recognized", zero = "Variable is out of range", word = "String failed assertion" }
  194. local changedT = {}
  195. changedT.new = function(key, value) changedT[#changedT+1] = {[key] = value} end
  196. changedT.getPair = function(i) for a, b in pairs(changedT[i]) do return a, b end end
  197. local function capitalize(text) return (string.upper(string.sub(text,1,1))..string.sub(text,2,-1)) end
  198. local function assert(condition, message, section) section = section or "[Blank]"; if condition then return condition else error("Error: "..message.."\nin section "..section, 0) end end
  199. local function checkNum(number, section) return assert(tonumber(number),errorT.num, section) end
  200. tArgs.checkStart = function(num) tArgs[tArgs[num]] = num end
  201. for i=1, #tArgs do tArgs.checkStart(i) end
  202.  
  203. --Check if it is a turtle
  204. if not turtle then
  205. print("This is not a turtle, you must be looking for the \"Companion Rednet Program\" \nCheck My forum thread for that")
  206. print("Press 'q' to quit, or any other key to start help ")
  207. if ({os.pullEvent("char")})[2] ~= "q" then tArgs.help = true else error("",0) end
  208. end
  209.  
  210.  
  211. if tArgs["help"] or tArgs["-help"] or tArgs["-?"] or tArgs["?"] then
  212. print("You have selected help, press any key to continue"); print("Use arrow keys to naviate, q to quit"); os.pullEvent("key")
  213. local pos = 1
  214. local key = 0
  215. while pos <= #help and key ~= keys.q do
  216. if pos < 1 then pos = 1 end
  217. screen(1,1)
  218. print(help[pos].title)
  219. for a=1, #help[pos] do print(help[pos][a]) end
  220. repeat
  221. _, key = os.pullEvent("key")
  222. until key == 200 or key == 208 or key == keys.q
  223. if key == 200 then pos = pos - 1 end
  224. if key == 208 then pos = pos + 1 end
  225. end
  226. error("",0)
  227. end
  228.  
  229. --Saving
  230. if tArgs["doBackup"] then doBackup = (string.lower(string.sub(tArgs[tArgs["doBackup"]+1],1,1)) ~= "f") end
  231. if tArgs["saveFile"] then saveFile = tArgs[tArgs["saveFile"]+1] or saveFile end
  232.  
  233. local restoreFound = false
  234. if tArgs["-restore"] or tArgs["-resume"] then
  235. restoreFound = fs.exists(saveFile)
  236. if restoreFound then
  237. fs.open(saveFile,"r")
  238. if file.readAll() ~= "" and file.close() then
  239. os.run(getfenv(1),saveFile)
  240. if gpsEnabled then --If it had saved gps coordinates
  241. print("Found GPS Start Coordinates")
  242. local currLoc = {gps.locate(gpsTimeout)}
  243. if #currLoc > 0 and #gpsStartPos > 0 and #gpsSecondPos > 0 then --Cover all the different positions I'm using
  244. print("GPS Position Successfully Read")
  245. if currLoc[1] == gpsStartPos[1] and currLoc[3] == gpsStartPos[3] then --X coord, y coord, z coord in that order
  246. xPos, yPos, zPos = 0,1,1
  247. if facing ~= 0 then turnTo(0) end
  248. print("Is at start")
  249. else
  250. if inverted then --yPos setting
  251. ------------------------------------------------FIX THIS
  252. end
  253. local function copyTable(tab) local toRet = {}; for a, b in pairs(tab) do toRet[a] = b end; return toRet end
  254. local a, b = copyTable(gpsStartPos), copyTable(gpsSecondPos) --For convenience
  255. if b[3] - a[3] == -1 then--If went north (-Z)
  256. a[1] = a[1] - 1 --Shift x one to west to create a "zero"
  257. xPos, zPos = -currLoc[3] + a[3], currLoc[1] + -a[1]
  258. elseif b[1] - a[1] == 1 then--If went east (+X)
  259. a[3] = a[3] - 1 --Shift z up one to north to create a "zero"
  260. xPos, zPos = currLoc[1] + -a[1], currLoc[3] + -a[3]
  261. elseif b[3] - a[3] == 1 then--If went south (+Z)
  262. a[1] = a[1] + 1 --Shift x one to east to create a "zero"
  263. xPos, zPos = currLoc[3] + a[3], -currLoc[1] + a[3]
  264. elseif b[1] - a[1] == -1 then--If went west (-X)
  265. a[3] = a[3] + 1 --Shift z down one to south to create a "zero"
  266. xPos, zPos = -currLoc[1] + a[1], -currLoc[3] + a[3]
  267. else
  268. print("Improper Coordinates")
  269. print("GPS Locate Failed, Using Standard Methods") ----Maybe clean this up a bit to use flags instead.
  270. end
  271. end
  272. print("X Pos: ",xPos)
  273. print("Y Pos: ",yPos)
  274. print("Z Pos: ",zPos)
  275. print("Facing: ",facing)
  276. else
  277. print("GPS Locate Failed, Using Standard Methods")
  278. end
  279. print("Restore File read successfully. Starting in 3"); sleep(3)
  280. end
  281. else
  282. originalFuel = checkFuel() --For use in logging. To see how much fuel is REALLY used
  283. end
  284. else
  285. fs.delete(saveFile)
  286. print("Restore file was empty, sorry, aborting")
  287. error("",0)
  288. end
  289. end
  290.  
  291. if not (tArgs["-DEFAULT"] or restoreFound) then
  292. --Just a note, the whole error checking section here was more of an
  293. --experiment than anything else. If its ever changed in the future,
  294. --it would probably be for the better.
  295. local section = "Dimensions"
  296. --Dimesnions
  297. if tArgs["-dim"] then local num = tArgs["-dim"];
  298. x = checkNum(tArgs[num + 1],section); z = checkNum(tArgs[num + 2],section); y = checkNum(tArgs[num + 3],section)
  299. else
  300. print("What dimensions?")
  301. print("")
  302. --This will protect from negatives, letters, and decimals
  303. term.write("Length: ")
  304. x = math.floor(math.abs(tonumber(io.read()) or x))
  305. term.write("Width: ")
  306. z = math.floor(math.abs(tonumber(io.read()) or z))
  307. term.write("Height: ")
  308. y = math.floor(math.abs(tonumber(io.read()) or y))
  309. end
  310. changedT.new("x",x); changedT.new("z",z); changedT.new("y",y)
  311. assert(x~=0, errorT.zero, section); assert(z~=0, errorT.zero, section); assert(y~=0, errorT.zero, section)
  312. assert(not(x == 1 and y == 1 and z == 1) ,"1, 1, 1 dosen't work well at all, try again", section)
  313. if not tArgs["-vanilla"] then
  314. --Invert
  315. if tArgs["-invert"] then
  316. inverted = (string.lower(string.sub(tArgs[tArgs["-invert"]+1] or "",1,1)) == "t") else
  317. term.write("Inverted? ")
  318. inverted = (string.lower(string.sub(io.read(),1,1)) == "y")
  319. end
  320. changedT.new("Inverted", inverted)
  321. --Rednet
  322. if supportsRednet then
  323. if tArgs["-rednet"] then
  324. rednetEnabled = (string.lower(string.sub(tArgs[tArgs["-rednet"]+1] or "",1,1)) == "t")
  325. else term.write("Rednet? "); rednetEnabled = (string.lower(string.sub(io.read(),1,1)) == "y")
  326. if (tArgs["-GPS"] or tArgs["-gps"]) and not restoreFound and not gpsEnabled then --The not gps enabled makes it so it is toggled
  327. gpsStartPos = {gps.locate(gpsTimeout)} --Stores position in array
  328. gpsEnabled = #gpsStartPos > 0 --Checks if location received properly
  329. changedT.new("GPS Location Services", gpsEnabled)
  330. end
  331. changedT.new("Rednet Enabled", rednetEnabled)
  332. end
  333. if tArgs["-sendChannel"] then
  334. channels.send = assert(tonumber(tArgs[tArgs["-sendChannel"]+1]), errorT.num)
  335. assert(channels.send > 0 and channels.send < 65535, errorT.zero)
  336. changedT.new("Send Channel",channels.send) end
  337. if tArgs["-receiveChannel"] then
  338. channels.receive = assert(tonumber(tArgs[tArgs["-receiveChannel"]+1]), errorT.num)
  339. assert(channels.receive > 0 and channels.receive < 65535 and channels.receive ~= channels.send, errorT.zero)
  340. changedT.new("Receive Channel",channels.receive) end
  341. end
  342. --Fuel
  343. if tArgs["-doRefuel"] then doRefuel = not doRefuel; changedT.new("Do Refuel",doRefuel) end
  344. if turtle.getFuelLevel() == "unlimited" then
  345. doCheckFuel = false
  346. else
  347. if tArgs["-doCheckFuel"] then
  348. doCheckFuel = (string.lower(string.sub(tArgs[tArgs["-doCheckFuel"]+1] or "",1,1)) == "t"); changedT.new("Do Check Fuel", doCheckFuel) end
  349. end
  350. if tArgs["-chest"] then
  351. dropSide = sides[tArgs[tArgs["-chest"]+1]] or dropSide; changedT.new("Chest Side",dropSide) end
  352. if tArgs["-fuelSafety"] then local loc = tArgs[tArgs["-fuelSafety"]+1]
  353. if fuelTable[loc] then
  354. fuelSafety = loc; changedT.new("Fuel Check Safety", fuelSafety)
  355. end
  356. end
  357. if tArgs["-enderChest"] then enderchestEnabled = true end --I know this doesn't have a default, or fancy toggling. This is just a hack for now.
  358. --Logging
  359. logging = (tArgs["-log"] and true) or logging --The and will set it to true if value, or its original value if nil
  360. if logging then changedT.new("Logging",logging) end
  361. if tArgs["-logFolder"] then logFolder = tArgs[tArgs["-logFolder"]+1] or logFolder; changedT.new("Log Folder",logFolder) end
  362. if tArgs["-logExtension"] then logExtension = tArgs[tArgs["-logExtension"]+1] or logExtension; changedT.new("Log Extension",logExtension) end
  363. --Misc
  364. if tArgs["-startY"] then
  365. startY = math.abs(math.floor(checkNum(tArgs[tArgs["-startY"]+1],"Start Y")))
  366. changedT.new("Start Y Position",startY) end
  367. assert(startY >= 0, errorT.zero, "StartY")
  368. if tArgs["-invCheckFreq"] then
  369. invCheckFreq = math.abs(math.floor(checkNum(tArgs[tArgs["-invCheckFreq"]+1],"Inventory Check Frequency")))
  370. changedT.new("Inventory Check Frequency",invCheckFreq) end
  371. assert(invCheckFreq ~= 0, errorT.zero, "Inventory Check Frequency")
  372. if tArgs["-keepOpen"] then
  373. keepOpen = math.abs(math.floor(checkNum(tArgs[tArgs["-keepOpen"]+1],"Open Slots")))
  374. changedT.new("Slots to keep open", keepOpen) end
  375. assert(keepOpen ~= 0 and keepOpen < 16, errorT.zero, "Open Slots")
  376. if tArgs["-ignoreResources"] then careAboutResources = false; changedT.new("Ignore Resources?", not careAboutResources) end
  377. if tArgs["-saveFile"] then saveFile = tArgs[tArgs["-saveFile"]+1] changedT.new("Save File", saveFile) end
  378. assert(#saveFile >= 2,errorT.word, "Save File")
  379. end; end --First end is for vanilla, second is for DEFAULT
  380.  
  381. local function saveProgress(extras) --Session persistence
  382. if doBackup then
  383. local toWrite = ""
  384. for a,b in pairs(getfenv(1)) do
  385. if type(b) == "string" then b = "\""..b.."\"" end
  386. if type(b) == "table" and a~="modem" then b = textutils.serialize(b) end
  387. if type(b) ~= "function" then
  388. toWrite = toWrite..a.." = "..tostring(b).."\n"
  389. end
  390. end
  391. toWrite = toWrite.."doCheckFuel = false\n" --It has already used fuel, so calculation unnesesary
  392. local file
  393. repeat
  394. file = fs.open(saveFile,"w")
  395. until file --WHY DOES IT SAY ATTEMPT TO INDEX NIL!!!
  396. file.write(toWrite)
  397. if type(extras) == "table" then
  398. for a, b in pairs(extras) do
  399. file.write(a.." = "..tostring(b))
  400. end
  401. end
  402. file.close()
  403. end
  404. end
  405.  
  406. local area = x*z
  407. local volume = x*y*z
  408. local lastHeight = y%3
  409. local dispY = y
  410. y = math.floor(y/3)*3
  411. local yMult = y/3 + math.ceil(lastHeight/2) --This is basically a smart y/3 for movement
  412. local moveVolume = (area * yMult) --Kept for display percent
  413. --Calculating Needed Fuel--
  414. local exStack = numberOfStacksPerRun --Expected stacks of items before full
  415. neededFuel = yMult * x * z + --This is volume it will run through
  416. dispY*(2+1/(64*exStack)) + --This is simplified and includes the y to get up times up and down + how many times it will drop stuff off
  417. (x+z)*(yMult + 1/(64*exStack)) --Simplified as well: It is getting to start of row plus getting to start of row how many times will drop off stuff
  418. --Original equation: x*z*y/3 + y * 2 + (x+z) * y/3 + (x + z + y) * (1/64*8)
  419. neededFuel = math.ceil(neededFuel)
  420.  
  421. --Getting Fuel
  422. if doCheckFuel and checkFuel() < neededFuel then
  423. neededFuel = neededFuel + fuelTable[fuelSafety] --For safety
  424. print("Not enough fuel")
  425. print("Current: ",checkFuel()," Needed: ",neededFuel)
  426. print("Starting SmartFuel...")
  427. sleep(2) --So they can read everything.
  428. term.clear()
  429. local oneFuel, neededFuelItems
  430. local currSlot = 0
  431. local function output(text, x, y) --For displaying fuel
  432. currX, currY = term.getCursorPos()
  433. term.setCursorPos(x,y)
  434. term.clearLine()
  435. term.write(text)
  436. term.setCursorPos(currX,currY)
  437. end
  438. local function roundTo(num, target) --For stacks of fuel
  439. if num >= target then return target elseif num < 0 then return 0 else return num end
  440. end
  441. local function updateScreen()
  442. output("Welcome to SmartFuel! Now Refueling...", 1,1)
  443. output("Currently taking fuel from slot "..currSlot,1,2)
  444. output("Current single fuel: "..tostring(oneFuel or 0),1,3)
  445. output("Current estimate of needed fuel: ",1,4)
  446. output("Single Items: "..math.ceil(neededFuelItems or 0),4,5)
  447. output("Stacks: "..math.ceil((neededFuelItems or 0) / 64),4,6)
  448. output("Needed Fuel: "..tostring(neededFuel),1,12)
  449. output("Current Fuel: "..tostring(checkFuel()),1,13)
  450. end
  451. while checkFuel() <= neededFuel do
  452. currSlot = currSlot + 1
  453. turtle.select(currSlot)
  454. updateScreen()
  455. while turtle.getItemCount(currSlot) == 0 do sleep(1.5) end
  456. repeat
  457. local previous = checkFuel()
  458. turtle.refuel(1)
  459. oneFuel = checkFuel() - previous
  460. updateScreen()
  461. until (oneFuel or 0) > 0 --Not an if to prevent errors if fuel taken out prematurely.
  462. neededFuelItems = (neededFuel - checkFuel()) / oneFuel
  463. turtle.refuel(roundTo(neededFuelItems, 64)) --Change because can only think about 64 at once.
  464. if turtle.getItemCount(roundTo(currSlot + 1, 16)) == 0 then --Resets if no more fuel
  465. currSlot = 0
  466. end
  467. neededFuelItems = (neededFuel - checkFuel()) / oneFuel
  468. end
  469. -- local neededFuel = moveVolume + (math.floor(volume / (64 * 8)) * (x+dispY+z)) --Standard move plus dropping off supplies
  470. -- --How many times come back to start| * If it were at the very far side
  471. -- neededFuel = neededFuel + fuelTable[fuelSafety]
  472. -- if neededFuel < 100 then neededFuel = 100; end
  473. -- if checkFuel() < neededFuel then
  474. -- screen(1,1)
  475. -- print("More Fuel Needed")
  476. -- print("Current Fuel: ",checkFuel()," Needed: ",neededFuel)
  477. -- print("Place fuel in Bottom Right")
  478. -- while turtle.getItemCount(16) == 0 do
  479. -- sleep(1)
  480. -- end
  481. -- turtle.select(16)
  482. -- while checkFuel() < neededFuel do
  483. -- if not turtle.refuel(1) then
  484. -- term.clearLine()
  485. -- print("Still too little fuel")
  486. -- term.clearLine()
  487. -- print("Insert more fuel to resume")
  488. -- while turtle.getItemCount(16) == 0 do
  489. -- sleep(1)
  490. -- end
  491. -- end
  492. -- local x,y = term.getCursorPos()
  493. -- print(checkFuel().." Fuel")
  494. -- term.setCursorPos(x,y)
  495. -- end
  496. -- print(checkFuel().." Units of Fuel")
  497. -- sleep(3)
  498. -- turtle.select(1)
  499. -- end
  500. end
  501. --Initial Rednet Handshake
  502. if rednetEnabled then
  503. screen(1,1)
  504. print("Rednet is Enabled")
  505. print("The Channel to open is "..channels.send)
  506. modem = peripheral.wrap("right")
  507. modem.open(channels.receive)
  508. local i = 0
  509. repeat
  510. local id = os.startTimer(3)
  511. i=i+1
  512. print("Sending Initial Message "..i)
  513. modem.transmit(channels.send, channels.receive, "{ 'Initial' }")
  514. local message
  515. repeat
  516. local event, idCheck, channel,_,locMessage, distance = os.pullEvent()
  517. message = locMessage
  518. until (event == "timer" and idCheck == id) or (event == "modem_message" and channel == channels.receive and message == channels.confirm)
  519. until message == channels.confirm
  520. connected = true
  521. print("Connection Confirmed!")
  522. sleep(1.5)
  523. end
  524. local function biometrics(sendChannel)
  525. local commands = { Confirm = "Confirm" }
  526. local toSend = { ["x"] = x, ["y"] = (y/3 + math.ceil(lastHeight/2)), ["z"] = z, --The y calc is weird...
  527. ["xPos"] = xPos, ["yPos"] = yPos, ["zPos"] = zPos,
  528. ["percent"] = percent, ["mined" ]= mined,
  529. ["fuel"] = checkFuel(), ["moved"] = moved,
  530. ["remainingBlocks"] = (volume-mined), ["ID"] = os.getComputerID(),
  531. ["isInPath"] = isInPath, --Whether it is going back to start
  532. ["volume"] = volume, ["area"] = area}
  533. modem.transmit(channels.send, channels.receive, textutils.serialize(toSend))
  534. id = os.startTimer(0.1)
  535. local event, message
  536. repeat
  537. local locEvent, idCheck, confirm, _, locMessage, distance = os.pullEvent()
  538. event, message = locEvent, locMessage
  539. until (event == "timer" and idCheck == id) or (event == "modem_message" and confirm == channels.receive)
  540. if event == "modem_message" then connected = true else connected = false end
  541. --Stuff to do for different commands
  542. end
  543. --Showing changes to settings
  544. screen(1,1)
  545. print("Your selected settings:")
  546. if #changedT == 0 then
  547. print("Completely Default")
  548. else
  549. for i=1, #changedT do
  550. local title, value = changedT.getPair(i)
  551. print(capitalize(title)..": ",value)
  552. end
  553. end
  554. print("\nStarting in 3"); sleep(1); print("2"); sleep(1); print("1"); sleep(1.5) --Dramatic pause at end
  555.  
  556.  
  557.  
  558. ----------------------------------------------------------------
  559. --Define ALL THE FUNCTIONS
  560. function display() --This is just the last screen that displays at the end
  561. screen(1,1)
  562. print("Total Blocks Mined: "..mined)
  563. print("Current Fuel Level: "..turtle.getFuelLevel())
  564. print("Cobble: "..totals.cobble)
  565. print("Usable Fuel: "..totals.fuel)
  566. print("Other: "..totals.other)
  567. if rednetEnabled then
  568. print("")
  569. print("Sent Stop Message")
  570. finalTable = {{["Mined: "] = mined}, {["Cobble: "] = totals.cobble}, {["Fuel: "] = totals.fuel},
  571. {["Other: "] = totals.other}, {["Fuel: "] = checkFuel()} }
  572. modem.transmit(channels.send,channels.receive,"stop")
  573. modem.transmit(channels.send,channels.receive,textutils.serialize(finalTable))
  574. modem.close(channels.receive)
  575. end
  576. if doBackup then fs.delete(saveFile) end
  577. end
  578. function updateDisplay() --Runs in Mine(), display information to the screen in a certain place
  579. screen(1,1)
  580. print("Blocks Mined")
  581. print(mined)
  582. print("Percent Complete")
  583. print(percent.."%")
  584. print("Fuel")
  585. print(checkFuel())
  586. -- screen(1,1)
  587. -- print("Xpos: ")
  588. -- print(xPos)
  589. -- print("RelXPos: ")
  590. -- print(relxPos)
  591. -- print("Z Pos: ")
  592. -- print(zPos)
  593. -- print("Y pos: ")
  594. -- print(yPos)
  595.  
  596. if rednetEnabled then
  597. screenLine(1,7)
  598. print("Connected: "..tostring(connected))
  599. end
  600. end
  601. function logMiningRun(textExtension, extras) --Logging mining runs
  602. if logging then
  603. local number
  604. if not fs.isDir(logFolder) then
  605. fs.delete(logFolder)
  606. fs.makeDir(logFolder)
  607. number = 1
  608. else
  609. local i = 0
  610. repeat
  611. i = i + 1
  612. until not fs.exists(logFolder.."/Quarry_Log_"..tostring(i)..(textExtension or ""))
  613. number = i
  614. end
  615. handle = fs.open(logFolder.."/Quarry_Log_"..tostring(number)..(textExtension or ""),"w")
  616. local function write(...)
  617. for a, b in ipairs({...}) do
  618. handle.write(tostring(b))
  619. end
  620. handle.write("\n")
  621. end
  622. write("Welcome to the Quarry Logs!")
  623. write("Entry Number: ",number)
  624. write("Dimensions (X Z Y): ",x," ",z," ", dispY)
  625. write("Blocks Mined: ", mined)
  626. write(" Cobble: ", totals.cobble)
  627. write(" Usable Fuel: ", totals.fuel)
  628. write(" Other: ",totals.other)
  629. write("Total Fuel Used: ", (originalFuel or (neededFuel + checkFuel()))- checkFuel()) --Protect against errors with some precision
  630. write("Expected Fuel Use: ", neededFuel)
  631. handle.close()
  632. end
  633. end
  634. function isFull(slots)
  635. slots = slots or 16
  636. local numUsed = 0
  637. sleep(0)
  638. for i=1, slots do
  639. if turtle.getItemCount(i) > 0 then numUsed = numUsed + 1 end
  640. end
  641. if numUsed >= slots then
  642. return true
  643. end
  644. return false
  645. end
  646. function dig(doAdd, func)
  647. doAdd = doAdd or true
  648. func = func or turtle.dig
  649. if func() then
  650. if doAdd then
  651. mined = mined + 1
  652. end
  653. return true
  654. end
  655. return false
  656. end
  657. function digUp(doAdd)
  658. return dig(doAdd,turtle.digUp)
  659. end
  660. function digDown(doAdd)
  661. return dig(doAdd,turtle.digDown)
  662. end
  663. function relativeXCalc()
  664. if rowCheck == "right" then relxPos = xPos else relxPos = (x-xPos)+1 end
  665. end
  666. function forward(doAdd)
  667. if doAdd == nil then doAdd = true end
  668. if turtle.forward() then
  669. if doAdd then
  670. moved = moved + 1
  671. end
  672. if facing == 0 then
  673. xPos = xPos + 1
  674. elseif facing == 1 then
  675. zPos = zPos + 1
  676. elseif facing == 2 then
  677. xPos = xPos - 1
  678. elseif facing == 3 then
  679. zPos = zPos - 1
  680. else
  681. error("Function forward, facing should be 0 - 3, got "..tostring(facing),2)
  682. end
  683. relativeXCalc()
  684. return true
  685. end
  686. return false
  687. end
  688. function up(sneak)
  689. sneak = sneak or 1
  690. if inverted and sneak == 1 then
  691. down(-1)
  692. else
  693. while not turtle.up() do
  694. if not digUp() then
  695. attackUp()
  696. sleep(0.5)
  697. end
  698. end
  699. yPos = yPos - sneak --Oh! I feel so clever
  700. end --This works because inverted :)
  701. end
  702. function down(sneak)
  703. sneak = sneak or 1
  704. local count = 0
  705. if inverted and sneak == 1 then
  706. up(-1)
  707. else
  708. while not turtle.down() do
  709. count = count + 1
  710. if not digDown() then
  711. attackDown()
  712. sleep(0.2)
  713. end
  714. if count > 20 then bedrock() end
  715. end
  716. yPos = yPos + sneak
  717. end
  718. end
  719. function right(num)
  720. num = num or 1
  721. for i=1, num do facingF(1); saveProgress(); turtle.turnRight() end
  722. end
  723. function left(num)
  724. num = num or 1
  725. for i=1, num do facingF(-1); saveProgress(); turtle.turnLeft() end
  726. end
  727. function attack(doAdd, func)
  728. doAdd = doAdd or true
  729. func = func or turtle.attack
  730. if func() then
  731. if doAdd then
  732. attacked = attacked + 1
  733. end
  734. return true
  735. end
  736. return false
  737. end
  738. function attackUp(doAdd)
  739. if inverted then
  740. return attack(doAdd, turtle.attackDown)
  741. else
  742. return attack(doAdd, turtle.attackUp)
  743. end
  744. end
  745. function attackDown(doAdd)
  746. if inverted then
  747. return attack(doAdd, turtle.attackUp)
  748. else
  749. return attack(doAdd, turtle.attackDown)
  750. end
  751. end
  752.  
  753.  
  754. function mine(doDigDown, doDigUp, outOfPath,doCheckInv) -- Basic Move Forward
  755. if doCheckInv == nil then doCheckInv = true end
  756. if doDigDown == nil then doDigDown = true end
  757. if doDigUp == nil then doDigUp = true end
  758. if outOfPath == nil then outOfPath = false end
  759. if inverted then
  760. doDigUp, doDigDown = doDigDown, doDigUp --Just switch the two if inverted
  761. end
  762. if doRefuel and checkFuel() <= fuelTable[fuelSafety]/2 then
  763. for i=1, 16 do
  764. if turtle.getItemCount(i) > 0 then
  765. turtle.select(i)
  766. if checkFuel() < 200 + fuelTable[fuelSafety] then
  767. turtle.refuel()
  768. end
  769. end
  770. end
  771. end
  772. local count = 0
  773. while not forward(not outOfPath) do
  774. sleep(0) --Calls coroutine.yield to prevent errors
  775. count = count + 1
  776. if not dig() then
  777. attack()
  778. end
  779. if count > 10 then
  780. attack()
  781. sleep(0.2)
  782. end
  783. if count > 50 then
  784. if turtle.getFuelLevel() == 0 then --Don't worry about inf fuel because I modified this function
  785. saveProgress({doCheckFuel = true})
  786. error("No more fuel",0)
  787. elseif yPos > (startY-7) then --If it is near bedrock
  788. bedrock()
  789. else --Otherwise just sleep for a bit to avoid sheeps
  790. sleep(1)
  791. end
  792. end
  793. end
  794. --Insanity Checking
  795. if xPos < 0 or xPos > x or zPos < 0 or zPos > z or yPos < 0 then
  796. print("Oops. Detected that quarry was outside of predefined boundaries.")
  797. print("Please go to my forum thread and report this with a short description of what happened")
  798. print("If you could also run \"pastebin put Civil_Quarry_Restore\" and give me that code it would be great")
  799. error("",0)
  800. end
  801. saveProgress(tab)
  802. if doDigUp then
  803. while turtle.detectUp() do
  804. sleep(0) --Calls coroutine.yield
  805. if not digUp() then
  806. attackUp()
  807. count = count + 1
  808. end
  809. if count > 50 and yPos > (startY-7) then --Same deal with bedrock as above
  810. bedrock()
  811. end
  812. end
  813. end
  814. if doDigDown then
  815. digDown()
  816. end
  817. percent = math.ceil(moved/moveVolume*100)
  818. updateDisplay()
  819. isInPath = (not outOfPath) --For rednet
  820. if doCheckInv and careAboutResources then
  821. if moved%invCheckFreq == 0 then
  822. if isFull(16-keepOpen) then dropOff() end
  823. end; end
  824. if rednetEnabled then biometrics() end
  825. end
  826. --Direction: Front = 0, Right = 1, Back = 2, Left = 3
  827. function facingF(num)
  828. facing = facing + num
  829. if facing > 3 then facing = 0 end
  830. if facing < 0 then facing = 3 end
  831. end
  832.  
  833. function turnTo(num, dir)
  834. num = num or facing
  835. dir = dir or "left"
  836. while facing ~= num do
  837. if dir == "left" then
  838. left()
  839. elseif dir == "right" then
  840. right()
  841. else
  842. error("TurnTo: Left or Right expected, got "..tosrting(dir))
  843. end
  844. end
  845. end
  846. function goto(x,z,y, toFace)
  847. --Will first go to desired z pos, then x pos, y pos varies
  848. x = x or 1; y = y or 1; z = z or 1; toFace = toFace or facing
  849. gotoX,gotoY,gotoZ,gotoFacing = xPos,yPos,zPos,facing --For use in session persistence
  850. if yPos > y then --Will go up first if below position
  851. while yPos~=y do up() end
  852. end
  853. if zPos > z then
  854. turnTo(3)
  855. elseif zPos < z then
  856. turnTo(1)
  857. end
  858. while zPos ~= z do mine(false,false,true,false) end
  859. if xPos > x then
  860. turnTo(2)
  861. elseif xPos < x then
  862. turnTo(0)
  863. end
  864. while xPos ~= x do mine(false,false,true,false) end
  865. if yPos < y then --Will go down after if above position
  866. while yPos~=y do down() end
  867. end
  868. turnTo(toFace,"right")
  869. saveProgress()
  870. gotoX,gotoY,gotoZ,gotoFacing = nil
  871. end
  872. function drop(side, final, allowSkip)
  873. side = sides[side] or "front" --The final number means that it will
  874. if final then final = 0 else final = 1 end --drop a whole stack at the end
  875. local allowSkip = allowSkip or (final == 0) --This will allow drop(side,t/f, rednetConnected)
  876. count()
  877. if doRefuel then
  878. for i=1, 16 do
  879. if slot[i][1] == 2 then
  880. turtle.select(i); turtle.refuel()
  881. end
  882. end
  883. turtle.select(1)
  884. end
  885. if side == "right" then turnTo(1) end
  886. if side == "left" then turnTo(3) end
  887. local whereDetect, whereDrop1, whereDropAll
  888. local _1 = slot[1][2] - final --All but one if final, all if not final
  889. if side == "top" then
  890. whereDetect = turtle.detectUp ; whereDrop = turtle.dropUp
  891. elseif side == "bottom" then
  892. whereDetect = turtle.detectDown ; whereDrop = turtle.dropDown
  893. else
  894. whereDetect = turtle.detect; whereDrop = turtle.drop
  895. end
  896. local function waitDrop(val) --This will just drop, but wait if it can't
  897. val = val or 64
  898. local try = 1
  899. while not whereDrop(val) do
  900. print("Chest Full, Try "..try)
  901. try = try + 1
  902. sleep(2)
  903. end
  904. end
  905. repeat
  906. local detected = whereDetect()
  907. if detected then
  908. waitDrop(_1)
  909. for i=2, 16 do
  910. if turtle.getItemCount(i) > 0 then
  911. turtle.select(i)
  912. waitDrop()
  913. end
  914. end
  915. elseif not allowSkip then
  916. print("Waiting for chest placement place a chest to continue")
  917. while not whereDetect() do
  918. sleep(1)
  919. end
  920. end
  921. until detected or allowSkip
  922. if not allowSkip then totals.cobble = totals.cobble - 1 end
  923. turtle.select(1)
  924. turnTo(0)
  925. end
  926. function dropOff() --Not local because called in mine()
  927. if enderchestEnabled then
  928. if turtle.getItemCount(16) == 1 then --Assuming there is an enderchest there
  929. right(); right();
  930. turtle.select(16)
  931. turtle.dig()
  932. turtle.place()
  933. drop("front",false)
  934. turtle.dig()
  935. turtle.select(1)
  936. right(); right();
  937. return true
  938. end
  939. end
  940. local currX,currZ,currY,currFacing = xPos, zPos, yPos, facing
  941. goto(0,1,1,2)
  942. drop(dropSide,false)
  943. mine(false,false,true, false)
  944. goto(1,1,1, 0)
  945. goto(currX,currZ,currY,currFacing)
  946. return true
  947. end
  948. function bedrock()
  949. if checkFuel() == 0 then error("No Fuel",0) end
  950. local origin = {x = xPos, y = yPos, z = zPos}
  951. print("Bedrock Detected")
  952. if turtle.detectUp() then
  953. print("Block Above")
  954. local var
  955. if facing == 0 then var = 2 elseif facing == 2 then var = 0 else error("Was facing left or right on bedrock") end
  956. goto(xPos,zPos,yPos,var)
  957. for i=1, relxPos do mine(true,true); end
  958. end
  959. goto(0,1,1,2)
  960. drop(dropSide, true)
  961. display()
  962. print("\nFound bedrock at these coordinates: ")
  963. print(origin.x," Was position in row\n",origin.z," Was row in layer\n",origin.y," Blocks down from start")
  964. error("",0)
  965. end
  966. -------------------------------------------------------------------------------------
  967. --Pre-Mining Stuff dealing with session persistence
  968. local doDigDown, doDigUp = (lastHeight ~= 1), false --Used in lastHeight
  969. if gpsEnabled and not restoreFound then --The initial locate is done in the arguments. This is so I can figure out what quadrant the turtle is in.
  970. gpsSecondPos = {gps.locate(gpsTimeout)} --Note: Does not run this if it has already been restarted.
  971. end
  972. if not restoreFound then --Regularly
  973. --Check if it is a mining turtle
  974. if not isMiningTurtle then
  975. local a, b = turtle.dig()
  976. if a then mined = mined + 1; isMiningTurtle = true
  977. elseif b == "Nothing to dig with" then
  978. print("This is not a mining turtle. To make a mining turtle, craft me together with a diamond pickaxe")
  979. error("",0)
  980. end
  981. end
  982. if y~= 0 then mine(false, false, true) down() end --Get into quarry
  983. else --restore found
  984. digUp(); digDown() --Get blocks missed before stopped
  985. if gotoX then
  986. goto(gotoX,gotoZ,gotoY,gotoFacing)
  987. end
  988. if (relxPos == x or relxPos == 1) and not gotoX and not (facing == 0 or facing == 2) then --If not in goto loop or facing standard direction
  989. if rowCheck == "right" then --If on odd rows (starting with first)
  990. turnTo(0) --Face the end so will turn properly
  991. else
  992. turnTo(2)
  993. end
  994. end
  995. end
  996. --Mining Loops
  997. turtle.select(1)
  998. while layersDone <= y do -------------Height---------
  999. moved = moved + 1 --To account for the first position in row as "moved"
  1000. if not restoreFound then rowCheck = "right"; relativeXCalc() end
  1001. while zPos <= z do -------------Width----------
  1002. while relxPos < x do ------------Length---------
  1003. mine()
  1004. end ---------------Length End-------
  1005. if zPos ~= z then
  1006. local func
  1007. if rowCheck == "right" and zPos ~= z then --Swithcing to next row
  1008. func = right; rowCheck = "left"; else func = left; rowCheck = "right" end --Which way to turn
  1009. func()
  1010. saveProgress({zPos = zPos + 1}) --This might make it so it will go back to proper spot z wise
  1011. mine()
  1012. func()
  1013. else break
  1014. end
  1015. end ---------------Width End--------
  1016. goto(1,1,yPos,0)
  1017. if yPos+1 ~= y then
  1018. for i=1, 3 do down() end
  1019. end
  1020. layersDone = layersDone + 3
  1021. restoreFound = false --This is done so that rowCheck works properly upon restore
  1022. end ---------------Height End-------
  1023. if lastHeight ~= 0 then ---------LAST ROW--------- (copied from above)
  1024. moved = moved + 1 --To account for the first position in row as "moved"
  1025. if y ~= 0 then --If the basic y == 2 or 1
  1026. for i=1, 2 do down() end
  1027. end
  1028. if not restoreFound then rowCheck = "right" end
  1029. relativeXCalc()
  1030. while zPos <= z do -------------Width----------
  1031. while relxPos < x do ------------Length---------
  1032. mine(doDigDown,doDigUp)
  1033. end ---------------Length End-------
  1034. if zPos ~= z then
  1035. local func
  1036. if rowCheck == "right" and zPos ~= z then --Swithcing to next row
  1037. func = right; rowCheck = "left"; else func = left; rowCheck = "right" end --Which way to turn
  1038. func()
  1039. mine(doDigDown,doDigUp)
  1040. func()
  1041. else break
  1042. end
  1043. end ---------------Width End--------
  1044. goto(1,1,yPos,0)
  1045. end
  1046. if doDigDown then
  1047. if inverted then
  1048. digUp()
  1049. else
  1050. digDown()
  1051. end
  1052. end
  1053. goto(0,1,1,2)
  1054.  
  1055. --Output to a chest or sit there
  1056. if enderchestEnabled and turtle.getItemCount(16) == 1 then
  1057. turtle.select(16)
  1058. turtle.place()
  1059. turtle.select(1)
  1060. end
  1061. drop(dropSide, true)
  1062.  
  1063. --Display was moved above to be used in bedrock function
  1064. display()
  1065. --Log current mining run
  1066. logMiningRun(logExtension)
  1067.  
  1068. --Cleanup
  1069. turtle.getFuelLevel = getFuel
Add Comment
Please, Sign In to add comment