2010-11-18 13:27:55 +00:00
|
|
|
-- common testing environment
|
2012-02-16 00:28:03 +01:00
|
|
|
require('posix')
|
2010-11-18 13:27:55 +00:00
|
|
|
|
|
|
|
-- escape codes to colorize output on terminal
|
2012-02-16 00:28:03 +01:00
|
|
|
local c1='\027[47;34m'
|
|
|
|
local c0='\027[0m'
|
2010-11-18 13:27:55 +00:00
|
|
|
|
|
|
|
---
|
|
|
|
-- writes colorized
|
|
|
|
--
|
|
|
|
function cwriteln(...)
|
|
|
|
io.write(c1, ...)
|
2012-02-16 00:28:03 +01:00
|
|
|
io.write(c0, '\n')
|
2010-11-18 13:27:55 +00:00
|
|
|
end
|
|
|
|
|
2010-11-24 20:34:56 +00:00
|
|
|
-----
|
|
|
|
-- initializes the pseudo random generator
|
2012-02-16 00:28:03 +01:00
|
|
|
-- if environemnt 'SEED' is set, use that as seed.
|
|
|
|
local seed = os.getenv('SEED') or os.time()
|
|
|
|
math.randomseed(seed)
|
|
|
|
cwriteln('random seed: ', seed)
|
2010-11-24 20:34:56 +00:00
|
|
|
|
2010-11-18 13:27:55 +00:00
|
|
|
-----
|
|
|
|
-- creates a tmp directory
|
2012-02-16 00:28:03 +01:00
|
|
|
--
|
2010-11-24 19:45:18 +00:00
|
|
|
-- @returns the name of the directory
|
|
|
|
--
|
2010-11-18 13:27:55 +00:00
|
|
|
function mktempd()
|
2010-11-21 00:11:00 +00:00
|
|
|
local f = io.popen('mktemp -td ltest.XXX', 'r')
|
2010-11-18 13:27:55 +00:00
|
|
|
local s = f:read('*a')
|
|
|
|
f:close()
|
|
|
|
s = s:gsub('[\n\r]+', ' ')
|
2012-02-16 00:28:03 +01:00
|
|
|
s = s:match('^%s*(.-)%s*$')
|
2010-11-18 13:27:55 +00:00
|
|
|
return s
|
|
|
|
end
|
|
|
|
|
2010-11-30 09:42:44 +00:00
|
|
|
-----
|
|
|
|
-- creates a tmp directory with the
|
|
|
|
-- typical lsyncd test architecture
|
2012-02-16 00:28:03 +01:00
|
|
|
--
|
2010-11-30 09:42:44 +00:00
|
|
|
-- @returns path of tmpdir
|
|
|
|
-- path of srcdir
|
|
|
|
-- path of trgdir
|
|
|
|
--
|
|
|
|
|
|
|
|
function mktemps()
|
2012-02-16 00:28:03 +01:00
|
|
|
local tdir = mktempd()..'/'
|
|
|
|
cwriteln('using ', tdir, ' as test root')
|
|
|
|
local srcdir = tdir..'src/'
|
|
|
|
local trgdir = tdir..'trg/'
|
2010-11-30 09:42:44 +00:00
|
|
|
posix.mkdir(srcdir)
|
|
|
|
posix.mkdir(trgdir)
|
|
|
|
return tdir, srcdir, trgdir
|
|
|
|
end
|
|
|
|
|
|
|
|
----
|
|
|
|
-- Writes a file with 'text' in it.
|
|
|
|
-- and adds a newline.
|
|
|
|
--
|
|
|
|
function writefile(filename, text)
|
2012-02-16 00:28:03 +01:00
|
|
|
local f = io.open(filename, 'w')
|
2010-11-30 09:42:44 +00:00
|
|
|
if not f then
|
2012-02-16 00:28:03 +01:00
|
|
|
cwriteln('Cannot open "'..filename..'" for writing.')
|
2010-11-30 09:42:44 +00:00
|
|
|
return false
|
|
|
|
end
|
|
|
|
f:write(text)
|
|
|
|
f:write('\n')
|
|
|
|
f:close()
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
|
2010-11-18 13:27:55 +00:00
|
|
|
-----
|
|
|
|
-- spawns a subprocess.
|
|
|
|
--
|
2010-11-18 16:33:48 +00:00
|
|
|
-- @returns the processes pid
|
|
|
|
--
|
2010-11-18 13:27:55 +00:00
|
|
|
function spawn(...)
|
2010-11-18 16:59:50 +00:00
|
|
|
args = {...}
|
2012-02-16 00:28:03 +01:00
|
|
|
cwriteln('spawning: ', table.concat(args, ' '))
|
2010-11-18 13:27:55 +00:00
|
|
|
local pid = posix.fork()
|
|
|
|
if pid < 0 then
|
2012-02-16 00:28:03 +01:00
|
|
|
cwriteln('Error, failed fork!')
|
2010-11-18 13:27:55 +00:00
|
|
|
os.exit(-1)
|
|
|
|
end
|
2010-11-18 16:59:50 +00:00
|
|
|
if pid == 0 then
|
2010-11-18 13:27:55 +00:00
|
|
|
posix.exec(...)
|
|
|
|
-- should not return
|
2012-02-16 00:28:03 +01:00
|
|
|
cwriteln('Error, failed to spawn: ', ...)
|
2010-11-18 13:27:55 +00:00
|
|
|
os.exit(-1);
|
|
|
|
end
|
|
|
|
return pid
|
|
|
|
end
|
|
|
|
|
2010-11-24 19:45:18 +00:00
|
|
|
-----
|
|
|
|
-- Makes a lot of random data
|
|
|
|
--
|
|
|
|
-- @param rootdir ... the directory to make data in
|
|
|
|
-- @param n ... roughly how much data action will done
|
|
|
|
--
|
2012-02-16 00:28:03 +01:00
|
|
|
function churn(rootdir, n)
|
2010-11-24 19:45:18 +00:00
|
|
|
-- all dirs created, indexed by integer and path
|
2012-02-16 00:28:03 +01:00
|
|
|
root = {name=''}
|
2010-11-24 19:45:18 +00:00
|
|
|
alldirs = {root}
|
|
|
|
dirsWithFileI = {}
|
|
|
|
dirsWithFileD = {}
|
|
|
|
|
|
|
|
-----
|
|
|
|
-- returns the name of a directory
|
|
|
|
--
|
|
|
|
-- name is internal recursive paramter, keep it nil.
|
|
|
|
--
|
|
|
|
local function dirname(dir, name)
|
2012-02-16 00:28:03 +01:00
|
|
|
name = name or ''
|
2010-11-24 19:45:18 +00:00
|
|
|
if not dir then
|
|
|
|
return name
|
|
|
|
end
|
2012-02-16 00:28:03 +01:00
|
|
|
return dirname(dir.parent, dir.name .. '/' .. name)
|
2010-11-24 19:45:18 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
-----
|
|
|
|
-- Picks a random dir.
|
|
|
|
--
|
|
|
|
local function pickDir(notRoot)
|
|
|
|
if notRoot then
|
|
|
|
if #alldirs <= 2 then
|
|
|
|
return nil
|
|
|
|
end
|
|
|
|
return alldirs[math.random(2, #alldirs)]
|
|
|
|
end
|
|
|
|
return alldirs[math.random(#alldirs)]
|
|
|
|
end
|
|
|
|
|
|
|
|
----
|
|
|
|
-- Picks a random file.
|
|
|
|
--
|
2012-02-16 00:28:03 +01:00
|
|
|
-- Returns 3 values:
|
2010-11-24 19:45:18 +00:00
|
|
|
-- * the directory
|
|
|
|
-- * the filename
|
|
|
|
-- * number of files in directory
|
|
|
|
--
|
|
|
|
local function pickFile()
|
|
|
|
-- picks the random directory
|
|
|
|
if #dirsWithFileI < 1 then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
local rdir = dirsWithFileI[math.random(1, #dirsWithFileI)]
|
|
|
|
if not rdir then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
-- counts the files in there
|
|
|
|
local c = 0
|
|
|
|
for name, _ in pairs(rdir) do
|
|
|
|
if #name == 2 then
|
|
|
|
c = c + 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- picks one file at random
|
|
|
|
local cr = math.random(1, c)
|
2012-02-16 00:28:03 +01:00
|
|
|
local fn
|
2010-11-24 19:45:18 +00:00
|
|
|
for name, _ in pairs(rdir) do
|
|
|
|
if #name == 2 then
|
|
|
|
-- filenames are 2 chars wide.
|
|
|
|
cr = cr - 1
|
|
|
|
if cr == 0 then
|
|
|
|
fn = name
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return rdir, fn, c
|
|
|
|
end
|
|
|
|
|
|
|
|
-----
|
|
|
|
-- Removes a reference to a file
|
|
|
|
--
|
|
|
|
-- @param dir --- directory reference
|
|
|
|
-- @param fn --- filename
|
|
|
|
-- @param c --- number of files in dir
|
|
|
|
--
|
|
|
|
local function rmFileReference(dir, fn, c)
|
|
|
|
dir[fn] = nil
|
|
|
|
if c == 1 then
|
|
|
|
-- if last file from origin dir, it has no files anymore
|
|
|
|
for i, v in ipairs(dirsWithFileI) do
|
2012-02-16 00:28:03 +01:00
|
|
|
if v == dir then
|
2010-11-24 19:45:18 +00:00
|
|
|
table.remove(dirsWithFileI, i)
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
dirsWithFileD[dir] = nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
----
|
2012-02-16 00:28:03 +01:00
|
|
|
-- possible randomized behaviour.
|
2010-11-24 19:45:18 +00:00
|
|
|
-- just gives it a pause
|
|
|
|
--
|
|
|
|
local function sleep()
|
2012-02-16 00:28:03 +01:00
|
|
|
cwriteln('..zzz..')
|
2010-11-24 19:45:18 +00:00
|
|
|
posix.sleep(1)
|
|
|
|
end
|
|
|
|
|
|
|
|
----
|
2012-02-16 00:28:03 +01:00
|
|
|
-- possible randomized behaviour.
|
2010-11-24 19:45:18 +00:00
|
|
|
-- creates a directory
|
|
|
|
--
|
|
|
|
local function mkdir()
|
|
|
|
-- chooses a random directory to create it into
|
|
|
|
local rdir = pickDir()
|
|
|
|
-- creates a new random one letter name
|
|
|
|
local nn = string.char(96 + math.random(26))
|
|
|
|
if not rdir[nn] then
|
|
|
|
local ndir = {
|
|
|
|
name = nn,
|
2012-02-16 00:28:03 +01:00
|
|
|
parent = rdir,
|
2010-11-24 19:45:18 +00:00
|
|
|
}
|
|
|
|
local dn = dirname(ndir)
|
|
|
|
rdir[nn] = dn
|
|
|
|
table.insert(alldirs, ndir)
|
2012-02-16 00:28:03 +01:00
|
|
|
cwriteln('mkdir '..rootdir..dn)
|
2010-11-24 19:45:18 +00:00
|
|
|
posix.mkdir(rootdir..dn)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
----
|
2012-02-16 00:28:03 +01:00
|
|
|
-- possible randomized behaviour.
|
|
|
|
-- Creates a file.
|
2010-11-24 19:45:18 +00:00
|
|
|
--
|
|
|
|
local function mkfile()
|
|
|
|
-- chooses a random directory to create it into
|
|
|
|
local rdir = pickDir()
|
|
|
|
-- creates a new random one letter name
|
|
|
|
local nn = 'f'..string.char(96 + math.random(26))
|
|
|
|
local fn = dirname(rdir) .. nn
|
2012-02-16 00:28:03 +01:00
|
|
|
cwriteln('mkfile '..rootdir..fn)
|
|
|
|
local f = io.open(rootdir..fn, 'w')
|
2010-11-24 19:45:18 +00:00
|
|
|
if f then
|
|
|
|
for i=1,10 do
|
|
|
|
f:write(string.char(96 + math.random(26)))
|
|
|
|
end
|
|
|
|
f:write('\n')
|
|
|
|
f:close()
|
|
|
|
rdir[nn]=true
|
|
|
|
if not dirsWithFileD[rdir] then
|
|
|
|
table.insert(dirsWithFileI, rdir)
|
|
|
|
dirsWithFileD[rdir]=true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
----
|
2012-02-16 00:28:03 +01:00
|
|
|
-- possible randomized behaviour,
|
|
|
|
-- moves a directory.
|
2010-11-24 19:45:18 +00:00
|
|
|
--
|
|
|
|
local function mvdir()
|
|
|
|
if #alldirs <= 2 then
|
|
|
|
return
|
|
|
|
end
|
2012-02-16 00:28:03 +01:00
|
|
|
-- chooses a random directory to move
|
2010-11-24 19:45:18 +00:00
|
|
|
local odir = pickDir(true)
|
|
|
|
-- chooses a random directory to move to
|
|
|
|
local tdir = pickDir()
|
|
|
|
|
|
|
|
-- makes sure tdir is not a subdir of odir
|
|
|
|
local dd = tdir
|
|
|
|
while dd do
|
|
|
|
if odir == dd then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
dd = dd.parent
|
|
|
|
end
|
|
|
|
-- origin name in the target dir already
|
|
|
|
if tdir[odir.name] ~= nil then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
local on = dirname(odir)
|
|
|
|
local tn = dirname(tdir)
|
2012-02-16 00:28:03 +01:00
|
|
|
cwriteln('mvdir ',rootdir,on,' -> ',rootdir,tn,odir.name)
|
2010-11-24 19:45:18 +00:00
|
|
|
os.rename(rootdir..on, rootdir..tn..odir.name)
|
|
|
|
odir.parent[odir.name] = nil
|
|
|
|
odir.parent = tdir
|
|
|
|
tdir[odir.name] = odir
|
|
|
|
end
|
|
|
|
|
|
|
|
----
|
2012-02-16 00:28:03 +01:00
|
|
|
-- possible randomized behaviour,
|
|
|
|
-- moves a file.
|
2010-11-24 19:45:18 +00:00
|
|
|
--
|
|
|
|
local function mvfile()
|
|
|
|
local odir, fn, c = pickFile()
|
|
|
|
if not odir then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
-- picks a directory with a file at random
|
|
|
|
-- picks a target directory at random
|
|
|
|
local tdir = pickDir()
|
|
|
|
local on = dirname(odir)
|
|
|
|
local tn = dirname(tdir)
|
2012-02-16 00:28:03 +01:00
|
|
|
cwriteln('mvfile ',rootdir,on,fn,' -> ',rootdir,tn,fn)
|
2010-11-24 19:45:18 +00:00
|
|
|
os.rename(rootdir..on..fn, rootdir..tn..fn)
|
|
|
|
rmFileReference(odir, fn, c)
|
|
|
|
|
|
|
|
tdir[fn] = true
|
|
|
|
if not dirsWithFileD[tdir] then
|
|
|
|
dirsWithFileD[tdir] = true
|
|
|
|
table.insert(dirsWithFileI, tdir)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
----
|
2012-02-16 00:28:03 +01:00
|
|
|
-- possible randomized behaviour,
|
|
|
|
-- removes a file.
|
2010-11-24 19:45:18 +00:00
|
|
|
--
|
|
|
|
local function rmfile()
|
|
|
|
local dir, fn, c = pickFile()
|
|
|
|
if dir then
|
|
|
|
local dn = dirname(dir)
|
2012-02-16 00:28:03 +01:00
|
|
|
cwriteln('rmfile ',rootdir,dn,fn)
|
2010-11-24 19:45:18 +00:00
|
|
|
posix.unlink(rootdir..dn..fn)
|
|
|
|
rmFileReference(dir, fn, c)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local dice = {
|
|
|
|
{ 10, sleep },
|
|
|
|
{ 20, mkfile },
|
|
|
|
{ 20, mkdir },
|
|
|
|
{ 20, mvdir },
|
|
|
|
{ 20, rmfile },
|
|
|
|
}
|
|
|
|
|
2012-02-16 00:28:03 +01:00
|
|
|
cwriteln('making random data')
|
2010-11-24 19:45:18 +00:00
|
|
|
local ndice = 0
|
|
|
|
for i, d in ipairs(dice) do
|
|
|
|
ndice = ndice + d[1]
|
|
|
|
d[1] = ndice
|
|
|
|
end
|
|
|
|
|
|
|
|
for ai=1,n do
|
2012-02-16 00:28:03 +01:00
|
|
|
-- throws a die what to do
|
2010-11-24 19:45:18 +00:00
|
|
|
local acn = math.random(ndice)
|
|
|
|
for i, d in ipairs(dice) do
|
|
|
|
if acn <= d[1] then
|
|
|
|
d[2]()
|
|
|
|
break
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
|