rewritten event combiner, now clearly not part of user script

This commit is contained in:
Axel Kittenberger 2010-11-28 23:39:18 +00:00
parent ed43c5c68a
commit fc70d4c3ed
3 changed files with 185 additions and 128 deletions

View File

@ -225,9 +225,169 @@ local Delay = (function()
return o return o
end end
-- public interface
return {new = new} return {new = new}
end)() end)()
-----
-- combines delays
--
local Combiner = (function()
----
-- new delay absorbed by old
--
local function abso(d1, d2)
log("Delay",d2.etype,":",d2.path," absorbed by ",
d1.etype,":",d1.path)
return "absorb"
end
----
-- new delay replaces the old one
--
local function repl(d1, d2)
log("Delay",d2.etype,":",d2.path," replaces ",
d1.etype,":",d1.path)
return "replace"
end
----
-- delays nullificate each other
--
local function null(d1, d2)
log("Delay",d2.etype,":",d2.path," nullifies ",
d1.etype,":",d1.path)
return "remove"
end
-----
-- TODO
--
local combineNoMove = {
Attrib = {Attrib=abso, Modify=repl, Create=repl, Delete=repl },
Modify = {Attrib=abso, Modify=abso, Create=repl, Delete=repl },
Create = {Attrib=abso, Modify=abso, Create=abso, Delete=null },
Delete = {Attrib=abso, Modify=abso, Create=repl, Delete=abso },
}
------
-- combines two delays
--
local function combine(d1, d2)
if d1.etype == "Blanket" then
-- everything is blocked by a blanket delay.
if d2.path2 then
log("Delay", d2.etype,":",d2.path,"->",d2.path2, "blocked by",
"Blanket event")
else
log("Delay", d2.etype,":",d2.path, "blocked by",
"Blanket event")
end
return "stack"
end
-- Two normal events
if d1.etype ~= "Move" and d2.etype ~= "Move" then
if d1.path == d2.path then
if d1.status == "active" then
return "stack"
end
return combineNoMove[d1.etype][d2.etype](d1, d2)
end
-- blocks events if one is a parent directory of another
if d1.path:byte(-1) == 47 and string.starts(d2.path, d1.path) or
d2.path:byte(-1) == 47 and string.starts(d1.path, d2.path)
then
return "stack"
end
return nil
end
-- Normal upon a Move
if d1.etype == "Move" and d2.etype ~= "Move" then
-- stacks the move if the from field could anyway be damaged
if d1.path == d2.path or
d2.path:byte(-1) == 47 and string.starts(d1.path, d2.path) or
d1.path:byte(-1) == 47 and string.starts(d2.path, d1.path)
then
log("Delay",d2.etype,":",d2.path," blocked by",
"Move :",d1.path,"->",d1.path2)
return "stack"
end
-- Event does something with the move destination
if d1.path2 == d2.path then
if d2.etype == "Delete" or d2.etype == "Create" then
if d1.status == "active" then
return "stack"
end
log("Delay",d2.etype,":",d2.path," turns ",
"Move :",d1.path,"->",d1.path2, " into ",
"Delete:",d1.path)
d1.etype = "Delete"
d1.path2 = nil
return "stack"
end
-- on "Attrib" or "Modify" simply wait for the move first
return "stack"
end
if d2.path :byte(-1) == 47 and string.starts(d1.path2, d2.path) or
d1.path2:byte(-1) == 47 and string.starts(d2.path, d1.path2)
then
log("Delay",d2.etype,":",d2.path," blocked by ",
"Move:",d1.path,"->",d1.path2)
return "stack"
end
return nil
end
-- Move upon a single event
if d1.etype ~= "Move" and d2.etype == "Move" then
if d1.path == d2.path or d1.path == d2.path2 or
d1.path :byte(-1) == 47 and string.starts(d2.path, d1.path) or
d1.path :byte(-1) == 47 and string.starts(d2.path2, d1.path) or
d2.path :byte(-1) == 47 and string.starts(d1.path, d2.path) or
d2.path2:byte(-1) == 47 and string.starts(d1.path, d2.path2)
then
log("Delay","Move:",d2.path,"->",d1.path2,
"splitted by ",d1.etype,":",d1.path)
return "split"
end
return nil
end
-- Move upon move
if d1.etype == "Move" and d2.etype == "Move" then
-- TODO combine moves,
if d1.path == d2.path or d1.path == d2.path2 or
d1.path2 == d2.path or d2.path2 == d2.path or
d1.path :byte(-1) == 47 and string.starts(d2.path, d1.path) or
d1.path :byte(-1) == 47 and string.starts(d2.path2, d1.path) or
d1.path2:byte(-1) == 47 and string.starts(d2.path, d1.path2) or
d1.path2:byte(-1) == 47 and string.starts(d2.path2, d1.path2) or
d2.path :byte(-1) == 47 and string.starts(d1.path, d2.path) or
d2.path :byte(-1) == 47 and string.starts(d1.path2, d2.path) or
d2.path2:byte(-1) == 47 and string.starts(d1.path, d2.path2) or
d2.path2:byte(-1) == 47 and string.starts(d1.path2, d2.path2)
then
log("Delay","Move:",d2.path,"->",d1.path2,
"splitted by Move:",d1.path,"->",d1.path2)
return "split"
end
return nil
end
error("reached impossible position")
end
-- public interface
return {combine = combine}
end)()
----- -----
-- User interface to grap events -- User interface to grap events
-- --
@ -1036,84 +1196,40 @@ local Sync = (function()
return return
end end
-- detects blocks and collapses by working from back until -- detects blocks and combos by working from back until
-- front through the fifo -- front through the fifo
InletControl.setSync(self)
local ne, ne2 = InletControl.d2e(nd)
local il = #self.delays -- last delay local il = #self.delays -- last delay
while il > 0 do while il > 0 do
-- get 'old' delay
local od = self.delays[il] local od = self.delays[il]
local oe, oe2 = InletControl.d2e(od) local ac = Combiner.combine(od, nd)
if oe.etype == "Blanket" then if ac then
-- everything is blocked by a blanket event. if ac == "remove" then
log("Delay", "Stacking ",nd.etype," upon blanket event.")
stack(od, nd)
table.insert(self.delays, nd)
return
end
-- this mini loop repeats the collapse a second
-- time for move events
local oel = oe
local nel = ne
while oel and nel do
local c = self.config.collapse(oel, nel, self.config)
if c == 0 then
-- events nullificate each ether
log("Delay",nd.etype," and ",od.etype," on ",path,
" nullified each other.")
od.etype = "None"
table.remove(self.delays, il) table.remove(self.delays, il)
return return
elseif c == 1 then elseif ac == "stack" then
log("Delay",nd.etype," is absored by event ",
od.etype," on ",path)
return
elseif c == 2 then
if od.etype ~= "Move" then
log("Delay",nd.etype," replaces event ",
od.etype," on ",path)
od.etype = nd.etype
if od.path ~= nd.path then
error("Cannot replace events with different paths")
end
else
log("Delay",nd.etype," turns a Move into delete of ",
od.path)
od.etype = "Delete"
od.path2 = nil
table.insert(self.delays, nd)
end
return
elseif c == 3 then
log("Delay", "Stacking ",nd.etype," upon ",
od.etype," on ",path)
stack(od, nd) stack(od, nd)
table.insert(self.delays, nd) table.insert(self.delays, nd)
return return
end elseif ac == "absorb" then
return
-- loops over all oe, oe2, ne, ne2 combos. elseif ac == "replace" then
if oel == oe and oe2 then od.etype = nd.etype
-- do another time for oe2 if present od.path = nd.path
oel = oe2 od.path2 = nd.path2
elseif nel == ne then return
-- do another time for ne2 if present elseif ac == "split" then
-- start with first oe delay(self, "Delete", time, path, nil)
nel = ne2 delay(self, "Create", time, path2, nil)
oel = oe return
else else
oel = false error("unknown result of combine()")
end end
end end
il = il - 1 il = il - 1
end end
log("Delay", "Registering ",nd.etype," on ",path) log("Delay", "New ",nd.etype,":",path)
-- there was no hit on collapse or it decided to stack. -- no block or combo
table.insert(self.delays, nd) table.insert(self.delays, nd)
end end
@ -1420,8 +1536,6 @@ local Syncs = (function()
end end
local defaultValues = { local defaultValues = {
'action', 'action',
'collapse',
'collapseTable',
'collect', 'collect',
'init', 'init',
'maxDelays', 'maxDelays',
@ -2913,9 +3027,9 @@ local default_rsyncssh = {
rsyncOps = "-lts", rsyncOps = "-lts",
----- -----
-- allow several processes -- allow processes
-- --
maxProcesses = 3, maxProcesses = 1,
------ ------
-- Let the core not split move event. -- Let the core not split move event.
@ -2954,62 +3068,6 @@ default = {
end end
end, end,
-----
-- Called to see if two events can be collapsed.
--
-- Default function uses the collapseTable.
--
-- @param event1 first event
-- @param event2 second event
-- @return -1 ... no interconnection
-- 0 ... drop both events.
-- 1 ... keep first event only
-- 2 ... keep second event only
-- 3 ... events block.
--
collapse = function(event1, event2, config)
if event1.path == event2.path then
if event1.status == "active" then
return 3
end
local e1 = event1.etype .. event1.move
local e2 = event2.etype .. event2.move
return config.collapseTable[e1][e2]
end
-----
-- Block events if one is a parent directory of another
--
if event1.isdir and string.starts(event2.path, event1.path) then
return 3
end
if event2.isdir and string.starts(event1.path, event2.path) then
return 3
end
return -1
end,
-----
-- Used by default collapse function.
-- Specifies how two event should be collapsed when here
-- horizontal event meets upon a vertical event.
-- values:
-- 0 ... nullification of both events.
-- 1 ... absorbtion of horizontal event.
-- 2 ... replace of vertical event.
-- 3 ... stack both events, vertical blocking horizonal.
-- 9 ... combines two move events.
--
collapseTable = {
Attrib = {Attrib=1, Modify=2, Create=2, Delete=2, MoveFr=3, MoveTo= 2},
Modify = {Attrib=1, Modify=1, Create=2, Delete=2, MoveFr=3, MoveTo= 2},
Create = {Attrib=1, Modify=1, Create=1, Delete=0, MoveFr=3, MoveTo= 2},
Delete = {Attrib=1, Modify=1, Create=3, Delete=1, MoveFr=3, MoveTo= 2},
MoveFr = {Attrib=3, Modify=3, Create=3, Delete=3, MoveFr=3, MoveTo= 3},
-- TODO MoveFr=9
MoveTo = {Attrib=3, Modify=3, Create=2, Delete=2, MoveFr=3, MoveTo= 2},
},
----- -----
-- Called when collecting a finished child process -- Called when collecting a finished child process

View File

@ -18,7 +18,7 @@ posix.mkdir(trgdir)
churn(srcdir, 10) churn(srcdir, 10)
local logs = {} local logs = {}
--logs = {"-log", "Inotify", "-log", "Exec" } logs = {"-log", "Delay" }
local pid = spawn("./lsyncd", "-nodaemon", "-delay", "5", local pid = spawn("./lsyncd", "-nodaemon", "-delay", "5",
"-rsync", srcdir, trgdir, unpack(logs)) "-rsync", srcdir, trgdir, unpack(logs))
@ -28,7 +28,7 @@ posix.sleep(1)
churn(srcdir, 100) churn(srcdir, 100)
cwriteln("waiting for Lsyncd to finish its jobs.") cwriteln("waiting for Lsyncd to finish its jobs.")
posix.sleep(30) posix.sleep(10)
cwriteln("killing the Lsyncd daemon") cwriteln("killing the Lsyncd daemon")
posix.kill(pid) posix.kill(pid)

View File

@ -18,8 +18,7 @@ posix.mkdir(trgdir)
churn(srcdir, 10) churn(srcdir, 10)
local logs = {} local logs = {}
--logs = {"-log", "Inotify", "-log", "Exec" } logs = {"-log", "Delay", "-log", "Exec"}
--logs = {"-log", "Delay"}
local pid = spawn("./lsyncd", "-nodaemon", "-delay", "5", local pid = spawn("./lsyncd", "-nodaemon", "-delay", "5",
"-rsyncssh", srcdir, "localhost", trgdir, "-rsyncssh", srcdir, "localhost", trgdir,
@ -31,7 +30,7 @@ posix.sleep(1)
churn(srcdir, 100) churn(srcdir, 100)
cwriteln("waiting for Lsyncd to finish its jobs.") cwriteln("waiting for Lsyncd to finish its jobs.")
posix.sleep(30) posix.sleep(10)
cwriteln("killing the Lsyncd daemon") cwriteln("killing the Lsyncd daemon")
posix.kill(pid) posix.kill(pid)