728 lines
15 KiB
Lua
728 lines
15 KiB
Lua
|
package.path = "/mod/?.lua;" .. package.path
|
||
|
local PriorityQueue = require "PriorityQueue"
|
||
|
|
||
|
function Set(list)
|
||
|
local set = {}
|
||
|
for _, l in ipairs(list) do set[l] = true end
|
||
|
return set
|
||
|
end
|
||
|
|
||
|
local yieldTime = 0
|
||
|
function yield()
|
||
|
if os.clock() - yieldTime > 2 then
|
||
|
sleep(0)
|
||
|
yieldTime = os.clock()
|
||
|
end
|
||
|
end
|
||
|
|
||
|
function arr(n, init)
|
||
|
local a = {}
|
||
|
for _ = 1, n do
|
||
|
table.insert(a, init)
|
||
|
end
|
||
|
return a
|
||
|
end
|
||
|
function arr2(x, y, init)
|
||
|
local a = {}
|
||
|
for _ = 1, y do
|
||
|
table.insert(a, arr(x, init))
|
||
|
end
|
||
|
return a
|
||
|
end
|
||
|
|
||
|
local dirMap = {
|
||
|
["0,-1"] = 0,
|
||
|
["0,1"] = 0.5,
|
||
|
["1,0"] = 0.25,
|
||
|
["-1,0"] = -0.25,
|
||
|
}
|
||
|
function direction(v)
|
||
|
local d = dirMap[v.x..","..v.z]
|
||
|
assert(d)
|
||
|
return d
|
||
|
end
|
||
|
|
||
|
function findDistances(points)
|
||
|
local dist = arr2(#points, #points, math.huge)
|
||
|
-- local prev = arr2(#points, #points)
|
||
|
|
||
|
for i = 1, #points do
|
||
|
for j = 1, #points do
|
||
|
if i ~= j then
|
||
|
dist[i][j] = (points[i] - points[j]):length()
|
||
|
-- prev[i][j] = i
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
for i = 1, #points do
|
||
|
dist[i][i] = 0
|
||
|
-- prev[i][i] = i
|
||
|
end
|
||
|
|
||
|
return dist
|
||
|
end
|
||
|
|
||
|
function groupOres(ores, dist, offset)
|
||
|
if not offset then
|
||
|
offset = 0
|
||
|
end
|
||
|
|
||
|
local groups = {}
|
||
|
for i, o in ipairs(ores) do
|
||
|
table.insert(groups, {
|
||
|
["type"] = o.name,
|
||
|
["blocks"] = {
|
||
|
[i] = vector.new(o.x, o.y, o.z),
|
||
|
},
|
||
|
})
|
||
|
end
|
||
|
|
||
|
local didJoin = true
|
||
|
while didJoin do
|
||
|
didJoin = false
|
||
|
for i, io in pairs(groups) do
|
||
|
for x, a in pairs(io.blocks) do
|
||
|
for j, jo in pairs(groups) do
|
||
|
if i ~= j and jo.type == io.type then
|
||
|
for y, b in pairs(jo.blocks) do
|
||
|
-- yield()
|
||
|
assert(a ~= b)
|
||
|
if dist[offset + x][offset + y] == 1 then
|
||
|
for z, p in pairs(jo.blocks) do
|
||
|
io.blocks[z] = p
|
||
|
end
|
||
|
groups[j] = nil
|
||
|
didJoin = true
|
||
|
break
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
local retGroups = {}
|
||
|
for _, g in pairs(groups) do
|
||
|
local v = vector.new(0, 0, 0)
|
||
|
local retBlocks = {}
|
||
|
for _, p in pairs(g.blocks) do
|
||
|
v = v + p
|
||
|
table.insert(retBlocks, p)
|
||
|
end
|
||
|
g.blocks = retBlocks
|
||
|
g.center = v / #g.blocks
|
||
|
assert(g.center)
|
||
|
table.insert(retGroups, g)
|
||
|
end
|
||
|
|
||
|
return retGroups
|
||
|
end
|
||
|
|
||
|
function pathThrough(points, dist, s, e)
|
||
|
-- local n = #points
|
||
|
|
||
|
-- dist[s][n] = 0
|
||
|
-- dist[n][s] = 0
|
||
|
-- dist[e][n] = 0
|
||
|
-- dist[n][e] = 0
|
||
|
|
||
|
-- for k = 1, #points do
|
||
|
-- for i = 1, #points do
|
||
|
-- for j = 1, #points do
|
||
|
-- if dist[i][j] > dist[i][k] + dist[k][j] then
|
||
|
-- dist[i][j] = dist[i][k] + dist[k][j]
|
||
|
-- prev[i][j] = prev[k][j]
|
||
|
-- end
|
||
|
-- end
|
||
|
-- yield()
|
||
|
-- end
|
||
|
-- end
|
||
|
|
||
|
-- local path = {e}
|
||
|
-- while s ~= e do
|
||
|
-- e = prev[s][e]
|
||
|
-- table.insert(path, 1, e)
|
||
|
-- end
|
||
|
|
||
|
-- return path
|
||
|
|
||
|
-- nearest neighbour
|
||
|
local unvisited = {}
|
||
|
local ui = nil
|
||
|
for i = 1, #points do
|
||
|
if not e or i ~= e then
|
||
|
table.insert(unvisited, i)
|
||
|
end
|
||
|
if i == s then
|
||
|
ui = #unvisited
|
||
|
end
|
||
|
end
|
||
|
assert(ui)
|
||
|
|
||
|
local visited = {}
|
||
|
local route = {}
|
||
|
while #unvisited ~= 1 do
|
||
|
local u = unvisited[ui]
|
||
|
table.remove(unvisited, ui)
|
||
|
|
||
|
local cDist = math.huge
|
||
|
local closest = nil
|
||
|
for ii = 1, #unvisited do
|
||
|
local i = unvisited[ii]
|
||
|
assert(i ~= u)
|
||
|
if dist[u][i] < cDist then
|
||
|
cDist = dist[u][i]
|
||
|
closest = ii
|
||
|
end
|
||
|
end
|
||
|
assert(closest)
|
||
|
table.insert(route, unvisited[closest])
|
||
|
ui = closest
|
||
|
end
|
||
|
if e then
|
||
|
table.insert(route, e)
|
||
|
end
|
||
|
|
||
|
return route
|
||
|
end
|
||
|
|
||
|
-- A*
|
||
|
function findPath(s, e)
|
||
|
local points = {}
|
||
|
function addPoint(p)
|
||
|
local k = p:tostring()
|
||
|
if not points[k] then
|
||
|
points[k] = p
|
||
|
end
|
||
|
return k
|
||
|
end
|
||
|
s = addPoint(s)
|
||
|
e = addPoint(e)
|
||
|
|
||
|
function h(i)
|
||
|
return (points[e] - points[i]):length()
|
||
|
end
|
||
|
|
||
|
function neighbors(k)
|
||
|
local p = points[k]
|
||
|
local ns = {
|
||
|
p + vector.new( 1, 0, 0),
|
||
|
p + vector.new( 0, 1, 0),
|
||
|
p + vector.new( 0, 0, 1),
|
||
|
p + vector.new(-1, 0, 0),
|
||
|
p + vector.new( 0, -1, 0),
|
||
|
p + vector.new( 0, 0, -1),
|
||
|
}
|
||
|
|
||
|
for i, n in ipairs(ns) do
|
||
|
ns[i] = addPoint(n)
|
||
|
end
|
||
|
return ns
|
||
|
end
|
||
|
|
||
|
local gScoreT = {}
|
||
|
gScoreT[s] = 0
|
||
|
function gScore(i)
|
||
|
if gScoreT[i] then
|
||
|
return gScoreT[i]
|
||
|
else
|
||
|
return math.huge
|
||
|
end
|
||
|
end
|
||
|
|
||
|
local fScoreT = {}
|
||
|
fScoreT[s] = h(s)
|
||
|
function fScore(i)
|
||
|
if fScoreT[i] then
|
||
|
return fScoreT[i]
|
||
|
else
|
||
|
return math.huge
|
||
|
end
|
||
|
end
|
||
|
|
||
|
local openSet = PriorityQueue(function(a, b)
|
||
|
return fScore(a) < fScore(b)
|
||
|
end)
|
||
|
openSet:enqueue(s, s)
|
||
|
|
||
|
local prev = {}
|
||
|
local found = false
|
||
|
while not openSet:empty() do
|
||
|
local current = openSet:dequeue()
|
||
|
if current == e then
|
||
|
found = true
|
||
|
break
|
||
|
end
|
||
|
|
||
|
local ns = neighbors(current)
|
||
|
for _, n in ipairs(ns) do
|
||
|
tentativeG = gScore(current) + 1
|
||
|
if tentativeG < gScore(n) then
|
||
|
prev[n] = current
|
||
|
gScoreT[n] = tentativeG
|
||
|
fScoreT[n] = tentativeG + h(n)
|
||
|
if not openSet:contains(n) then
|
||
|
openSet:enqueue(n, n)
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
assert(found)
|
||
|
|
||
|
local path = {points[e]}
|
||
|
local current = e
|
||
|
while prev[current] do
|
||
|
current = prev[current]
|
||
|
table.insert(path, 1, points[current])
|
||
|
end
|
||
|
return path
|
||
|
end
|
||
|
|
||
|
local ORES = Set{
|
||
|
"minecraft:coal_ore",
|
||
|
"minecraft:deepslate_coal_ore",
|
||
|
"minecraft:iron_ore",
|
||
|
"minecraft:deepslate_iron_ore",
|
||
|
"minecraft:copper_ore",
|
||
|
"minecraft:deepslate_copper_ore",
|
||
|
"minecraft:gold_ore",
|
||
|
"minecraft:deepslate_gold_ore",
|
||
|
"minecraft:redstone_ore",
|
||
|
"minecraft:deepslate_redstone_ore",
|
||
|
"minecraft:emerald_ore",
|
||
|
"minecraft:deepslate_emerald_ore",
|
||
|
"minecraft:lapis_ore",
|
||
|
"minecraft:deepslate_lapis_ore",
|
||
|
"minecraft:diamond_ore",
|
||
|
"minecraft:deepslate_diamond_ore",
|
||
|
|
||
|
"create:ochrum",
|
||
|
"create:zinc_ore",
|
||
|
"create:deepslate_zinc_ore",
|
||
|
"create_new_age:thorium_ore",
|
||
|
|
||
|
"powah:deepslate_uraninite_ore_poor",
|
||
|
"powah:deepslate_uraninite_ore",
|
||
|
"powah:deepslate_uraninite_ore_dense",
|
||
|
"powah:uraninite_ore_poor",
|
||
|
"powah:uraninite_ore",
|
||
|
"powah:uraninite_ore_dense",
|
||
|
|
||
|
-- "minecraft:diamond_ore",
|
||
|
-- "minecraft:netherrack",
|
||
|
}
|
||
|
local BLACKLIST = Set{
|
||
|
"minecraft:diamond_pickaxe",
|
||
|
"computercraft:wireless_modem_advanced",
|
||
|
"advancedperipherals:geo_scanner",
|
||
|
"enderchests:ender_chest",
|
||
|
"functionalstorage:ender_drawer",
|
||
|
"advancedperipherals:chunk_controller",
|
||
|
}
|
||
|
local FUEL = "minecraft:coal"
|
||
|
local CHANNEL = 1337
|
||
|
|
||
|
settings.define("mine.name", {
|
||
|
description = "Miner name",
|
||
|
default = "DiggyBoi",
|
||
|
type = "string",
|
||
|
})
|
||
|
|
||
|
function selectItem(item, nbt)
|
||
|
for i = 1, 16 do
|
||
|
info = turtle.getItemDetail(i)
|
||
|
if (not item and not info) or (info and info.name == item and (not nbt or item.nbt == nbt)) then
|
||
|
turtle.select(i)
|
||
|
return i
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
|
||
|
Miner = {}
|
||
|
function Miner.new()
|
||
|
local self = setmetatable({}, { __index = Miner })
|
||
|
self.name = settings.get("mine.name")
|
||
|
self.equipped = {
|
||
|
["left"] = {
|
||
|
["last"] = "NONE",
|
||
|
["lastNbt"] = nil,
|
||
|
["fun"] = turtle.equipLeft,
|
||
|
},
|
||
|
["right"] = {
|
||
|
["last"] = "NONE",
|
||
|
["lastNbt"] = nil,
|
||
|
["fun"] = turtle.equipRight,
|
||
|
},
|
||
|
}
|
||
|
|
||
|
-- Get upgrades to a known state by "equipping" nothing
|
||
|
self:equip(nil, "left")
|
||
|
self:equip(nil, "right")
|
||
|
|
||
|
self:equip("advancedperipherals:geo_scanner", "right")
|
||
|
self.scanner = peripheral.wrap("right")
|
||
|
|
||
|
self:equip("computercraft:wireless_modem_advanced", "right")
|
||
|
self.modem = peripheral.wrap("right")
|
||
|
self.modem.open(CHANNEL)
|
||
|
|
||
|
self.pos = vector.new(0, 0, 0)
|
||
|
self:doGPS(true)
|
||
|
|
||
|
self:equip("advancedperipherals:chunk_controller", "left")
|
||
|
|
||
|
self.i = 1
|
||
|
return self
|
||
|
end
|
||
|
|
||
|
function Miner:equip(item, side, nbt)
|
||
|
local e = self.equipped[side]
|
||
|
if item == e.last and nbt == e.lastNbt then
|
||
|
return
|
||
|
end
|
||
|
|
||
|
assert(selectItem(item, nbt))
|
||
|
e.fun()
|
||
|
e.last = item
|
||
|
e.lastNbt = nbt
|
||
|
end
|
||
|
|
||
|
function Miner:sendMessage(type, msg)
|
||
|
self:equip("computercraft:wireless_modem_advanced", "right")
|
||
|
msg["type"] = type
|
||
|
msg["name"] = self.name
|
||
|
msg["pos"] = self.absolutePos
|
||
|
self.modem.transmit(CHANNEL, CHANNEL, msg)
|
||
|
end
|
||
|
|
||
|
function Miner:dig(dir)
|
||
|
self:equip("minecraft:diamond_pickaxe", "right")
|
||
|
if dir == "up" then
|
||
|
assert(turtle.digUp())
|
||
|
elseif dir == "forward" then
|
||
|
assert(turtle.dig())
|
||
|
elseif dir == "down" then
|
||
|
assert(turtle.digDown())
|
||
|
end
|
||
|
end
|
||
|
|
||
|
function Miner:xchgItems()
|
||
|
local slots = {}
|
||
|
local emptySlots = 0
|
||
|
for i = 1, 16 do
|
||
|
local info = turtle.getItemDetail(i)
|
||
|
if info then
|
||
|
assert(info.name)
|
||
|
if not BLACKLIST[info.name] then
|
||
|
slots[i] = info
|
||
|
end
|
||
|
else
|
||
|
emptySlots = emptySlots + 1
|
||
|
end
|
||
|
end
|
||
|
|
||
|
if emptySlots >= 2 and turtle.getFuelLevel() >= 1000 then
|
||
|
return
|
||
|
end
|
||
|
|
||
|
if turtle.detectDown() then
|
||
|
self:dig("down")
|
||
|
end
|
||
|
|
||
|
if emptySlots < 2 then
|
||
|
selectItem("enderchests:ender_chest")
|
||
|
assert(turtle.placeDown())
|
||
|
|
||
|
for i, info in pairs(slots) do
|
||
|
assert(turtle.select(i))
|
||
|
assert(turtle.dropDown())
|
||
|
end
|
||
|
self:dig("down")
|
||
|
end
|
||
|
|
||
|
if turtle.getFuelLevel() < 1000 then
|
||
|
selectItem("functionalstorage:ender_drawer")
|
||
|
assert(turtle.placeDown())
|
||
|
assert(turtle.suckDown())
|
||
|
assert(turtle.refuel())
|
||
|
self:dig("down")
|
||
|
end
|
||
|
end
|
||
|
|
||
|
function Miner:safeMove(dir)
|
||
|
if dir == "up" then
|
||
|
while turtle.detectUp() do
|
||
|
self:dig("up")
|
||
|
end
|
||
|
assert(turtle.up())
|
||
|
elseif dir == "forward" then
|
||
|
while turtle.detect() do
|
||
|
self:dig("forward")
|
||
|
end
|
||
|
assert(turtle.forward())
|
||
|
else
|
||
|
if turtle.detectDown() then
|
||
|
self:dig("down")
|
||
|
end
|
||
|
assert(turtle.down())
|
||
|
end
|
||
|
end
|
||
|
|
||
|
function Miner:doGPS(orientation)
|
||
|
local x, y, z = gps.locate()
|
||
|
assert(x)
|
||
|
self.absolutePos = vector.new(x, y, z)
|
||
|
|
||
|
if orientation then
|
||
|
assert(turtle.forward())
|
||
|
x, y, z = gps.locate()
|
||
|
assert(x)
|
||
|
self.dir = vector.new(x, y, z) - self.absolutePos
|
||
|
assert(turtle.back())
|
||
|
end
|
||
|
end
|
||
|
|
||
|
function Miner:findOres(radius)
|
||
|
self:equip("advancedperipherals:geo_scanner", "right")
|
||
|
local info, err = self.scanner.scan(radius)
|
||
|
assert(info, err)
|
||
|
|
||
|
local found = {}
|
||
|
for _, b in ipairs(info) do
|
||
|
-- only ores and avoid bedrock
|
||
|
if ORES[b.name] and self.absolutePos.y + b.y >= -58 then
|
||
|
table.insert(found, b)
|
||
|
end
|
||
|
end
|
||
|
return found
|
||
|
end
|
||
|
|
||
|
function Miner:faceDir(newDir)
|
||
|
local deltaDir = direction(newDir) - direction(self.dir)
|
||
|
if deltaDir == 0.25 or deltaDir == -0.75 then
|
||
|
assert(turtle.turnRight())
|
||
|
self.dir = newDir
|
||
|
elseif deltaDir == -0.25 or deltaDir == 0.75 then
|
||
|
assert(turtle.turnLeft())
|
||
|
self.dir = newDir
|
||
|
elseif deltaDir == 0.5 or deltaDir == -0.5 then
|
||
|
assert(turtle.turnRight())
|
||
|
assert(turtle.turnRight())
|
||
|
self.dir = newDir
|
||
|
elseif deltaDir == 0 then
|
||
|
-- nothing to do :)
|
||
|
else
|
||
|
assert(false, "invalid delta dir "..deltaDir)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
function Miner:navigateThrough(path)
|
||
|
for _, p in ipairs(path) do
|
||
|
-- print("move from to", self.pos, p)
|
||
|
local delta = p - self.pos
|
||
|
assert(delta:length() ~= 0)
|
||
|
-- print("doing", delta)
|
||
|
if delta:length() ~= 1 then
|
||
|
-- print("path finding between points")
|
||
|
local path = findPath(self.pos, p)
|
||
|
table.remove(path, 1)
|
||
|
table.remove(path, #path)
|
||
|
assert(#path ~= 0)
|
||
|
|
||
|
self:navigateThrough(path)
|
||
|
delta = p - self.pos
|
||
|
end
|
||
|
assert(delta:length() == 1)
|
||
|
|
||
|
local moveDir = nil
|
||
|
if delta.y == 1 then
|
||
|
moveDir = "up"
|
||
|
elseif delta.y == -1 then
|
||
|
moveDir = "down"
|
||
|
else
|
||
|
self:faceDir(delta)
|
||
|
moveDir = "forward"
|
||
|
end
|
||
|
|
||
|
self:safeMove(moveDir)
|
||
|
self.pos = p
|
||
|
self.absolutePos = self.absolutePos + delta
|
||
|
end
|
||
|
end
|
||
|
|
||
|
function Miner:mineOres(radius)
|
||
|
local startingDir = self.dir
|
||
|
self.pos = vector.new(0, 0, 0)
|
||
|
|
||
|
local ores = self:findOres(radius)
|
||
|
local orePoints = {}
|
||
|
for _, b in ipairs(ores) do
|
||
|
table.insert(orePoints, vector.new(b.x, b.y, b.z))
|
||
|
end
|
||
|
local veins = groupOres(ores, findDistances(orePoints))
|
||
|
print("Found "..#ores.." ores ("..#veins.." veins)")
|
||
|
|
||
|
local veinsPoints = {
|
||
|
self.pos,
|
||
|
self.dir * radius * 2,
|
||
|
}
|
||
|
for _, v in ipairs(veins) do
|
||
|
table.insert(veinsPoints, v.center)
|
||
|
end
|
||
|
|
||
|
local veinsPath = pathThrough(veinsPoints, findDistances(veinsPoints), 1, 2)
|
||
|
-- strip out end point
|
||
|
table.remove(veinsPath, #veinsPath)
|
||
|
|
||
|
local veinsSummary = {}
|
||
|
for _, i in ipairs(veinsPath) do
|
||
|
local v = veins[i - 2]
|
||
|
table.insert(veinsSummary, {
|
||
|
pos = self.absolutePos + v.center,
|
||
|
type = v.type,
|
||
|
count = #v.blocks,
|
||
|
})
|
||
|
end
|
||
|
self:sendMessage("iterationStart", {
|
||
|
i = self.i,
|
||
|
oreCount = #ores,
|
||
|
veins = veinsSummary,
|
||
|
})
|
||
|
|
||
|
for pi, i in ipairs(veinsPath) do
|
||
|
self:xchgItems()
|
||
|
|
||
|
local vein = veins[i - 2]
|
||
|
local closest = nil
|
||
|
local cDist = math.huge
|
||
|
for _, b in ipairs(vein.blocks) do
|
||
|
local d = (b - self.pos):length()
|
||
|
if d < cDist then
|
||
|
closest = b
|
||
|
cDist = d
|
||
|
end
|
||
|
end
|
||
|
assert(closest)
|
||
|
|
||
|
local pathToStart = findPath(self.pos, closest)
|
||
|
table.remove(pathToStart, 1)
|
||
|
table.remove(pathToStart, #pathToStart)
|
||
|
|
||
|
print("Moving to vein of " .. #vein.blocks .. " " .. vein.type .. " " .. cDist .. " blocks away starting at", closest)
|
||
|
self:navigateThrough(pathToStart)
|
||
|
|
||
|
local veinPoints = {self.pos}
|
||
|
for _, b in ipairs(vein.blocks) do
|
||
|
table.insert(veinPoints, b)
|
||
|
end
|
||
|
|
||
|
local veinPathI = pathThrough(veinPoints, findDistances(veinPoints), 1)
|
||
|
local veinPath = {}
|
||
|
for _, i in ipairs(veinPathI) do
|
||
|
table.insert(veinPath, veinPoints[i])
|
||
|
end
|
||
|
|
||
|
print("Digging through vein (" .. #vein.blocks .. " blocks)")
|
||
|
self:sendMessage("veinStart", {
|
||
|
i = pi,
|
||
|
total = #veins,
|
||
|
oreType = vein.type,
|
||
|
oreCount = #vein.blocks,
|
||
|
})
|
||
|
self:navigateThrough(veinPath)
|
||
|
end
|
||
|
|
||
|
self:navigateThrough({veinsPoints[2]})
|
||
|
self:faceDir(startingDir)
|
||
|
self.i = self.i + 1
|
||
|
end
|
||
|
|
||
|
local miner = Miner.new()
|
||
|
while true do
|
||
|
miner:mineOres(16)
|
||
|
end
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
-- test code, please ignore
|
||
|
assert(false)
|
||
|
|
||
|
-- local f, err = fs.open("points.csv", "w")
|
||
|
-- assert(f, err)
|
||
|
|
||
|
local points = {
|
||
|
vector.new(0, 0, 0),
|
||
|
vector.new(0, 0, -16),
|
||
|
}
|
||
|
for _, b in ipairs(ores) do
|
||
|
-- print("found " .. b.name .. " at " .. b.x .. ", " .. b.y .. ", " .. b.z)
|
||
|
table.insert(points, vector.new(b.x, b.y, b.z))
|
||
|
end
|
||
|
print("found "..#ores.." ores")
|
||
|
|
||
|
local dist = findDistances(points)
|
||
|
local groups = groupOres(ores, dist, 2)
|
||
|
|
||
|
for _, g in ipairs(groups) do
|
||
|
print("group of "..#g.blocks.." "..g.type.." at", g.center)
|
||
|
end
|
||
|
print(#groups, "groups of ore found")
|
||
|
|
||
|
local gPoints = {
|
||
|
vector.new(0, 0, 0),
|
||
|
vector.new(0, 0, -16),
|
||
|
}
|
||
|
for _, g in ipairs(groups) do
|
||
|
table.insert(gPoints, g.center)
|
||
|
end
|
||
|
|
||
|
local gDist = findDistances(gPoints)
|
||
|
local gPath = pathThrough(gPoints, gDist, 1, 2)
|
||
|
for _, i in ipairs(gPath) do
|
||
|
local p = gPoints[i]
|
||
|
f.write(p.x..","..p.y..","..p.z.."\n")
|
||
|
end
|
||
|
|
||
|
f.write("MARK\n")
|
||
|
|
||
|
-- print(points[1], gPoints[gPath[1]])
|
||
|
-- local pathTo = findPath(points[1], gPoints[gPath[1]])
|
||
|
-- for _, p in ipairs(pathTo) do
|
||
|
-- -- local p = pathTo[i]
|
||
|
-- -- print(p)
|
||
|
-- f.write(p.x..","..p.y..","..p.z.."\n")
|
||
|
-- end
|
||
|
|
||
|
miner:mineOres(groups)
|
||
|
|
||
|
-- table.remove(pathTo, 1)
|
||
|
-- table.remove(pathTo, #pathTo)
|
||
|
-- miner:navigateThrough(pathTo)
|
||
|
|
||
|
-- f.write("MARK\n")
|
||
|
|
||
|
-- local allPath = pathThrough(points, dist, 1, 2)
|
||
|
-- for _, i in ipairs(allPath) do
|
||
|
-- local p = points[i]
|
||
|
-- -- print(points[i])
|
||
|
-- f.write(p.x..","..p.y..","..p.z.."\n")
|
||
|
-- end
|
||
|
|
||
|
f.close()
|
||
|
|
||
|
-- local f = vector.new(1, 0, 0)
|
||
|
-- local p = vector.new(0, -1, 7)
|
||
|
-- local t = vector.new(-1, -1, 7)
|
||
|
|
||
|
-- print(direction(t - p) - direction(f))
|