Merge 0.10->trunk

This commit is contained in:
Kim Alvefur 2017-03-28 20:14:35 +02:00
commit f389cff7ea
4 changed files with 74 additions and 76 deletions

View file

@ -7,7 +7,7 @@
--
local want_pposix_version = "0.3.6";
local want_pposix_version = "0.4.0";
local pposix = assert(require "util.pposix");
if pposix._VERSION ~= want_pposix_version then

View file

@ -135,7 +135,7 @@ dependencies.log_warnings();
-- Switch away from root and into the prosody user --
local switched_user, current_uid;
local want_pposix_version = "0.3.6";
local want_pposix_version = "0.4.0";
local ok, pposix = pcall(require, "util.pposix");
if ok and pposix then

View file

@ -13,7 +13,7 @@
* POSIX support functions for Lua
*/
#define MODULE_VERSION "0.3.6"
#define MODULE_VERSION "0.4.0"
#if defined(__linux__)
@ -730,62 +730,66 @@ int lc_meminfo(lua_State *L) {
}
#endif
/* File handle extraction blatantly stolen from
* https://github.com/rrthomas/luaposix/blob/master/lposix.c#L631
* */
/*
* Append some data to a file handle
* Attempt to allocate space first
* Truncate to original size on failure
*/
int lc_atomic_append(lua_State *L) {
int err;
size_t len;
int lc_fallocate(lua_State *L) {
int ret;
off_t offset, len;
FILE *f = *(FILE **) luaL_checkudata(L, 1, LUA_FILEHANDLE);
const char *data = luaL_checklstring(L, 2, &len);
if(f == NULL) {
return luaL_error(L, "attempt to use a closed file");
}
offset = luaL_checkinteger(L, 2);
len = luaL_checkinteger(L, 3);
off_t offset = ftell(f);
#if defined(__linux__)
errno = 0;
ret = fallocate(fileno(f), FALLOC_FL_KEEP_SIZE, offset, len);
/* Try to allocate space without changing the file size. */
if((err = fallocate(fileno(f), FALLOC_FL_KEEP_SIZE, offset, len))) {
if(errno != 0) {
/* Some old versions of Linux apparently use the return value instead of errno */
err = errno;
}
switch(err) {
case ENOSYS: /* Kernel doesn't implement fallocate */
case EOPNOTSUPP: /* Filesystem doesn't support it */
/* Ignore and proceed to try to write */
break;
if(ret == 0) {
lua_pushboolean(L, 1);
return 1;
case ENOSPC: /* No space left */
default: /* Other issues */
lua_pushnil(L);
lua_pushstring(L, strerror(err));
lua_pushinteger(L, err);
return 3;
}
}
/* Some old versions of Linux apparently use the return value instead of errno */
if(errno == 0) {
errno = ret;
}
if(errno != ENOSYS && errno != EOPNOTSUPP) {
lua_pushnil(L);
lua_pushstring(L, strerror(errno));
return 2;
}
#endif
ret = posix_fallocate(fileno(f), offset, len);
if(ret == 0) {
lua_pushboolean(L, 1);
return 1;
} else {
lua_pushnil(L);
lua_pushstring(L, strerror(ret));
/* posix_fallocate() can leave a bunch of NULs at the end, so we cut that
* this assumes that offset == length of the file */
if(ftruncate(fileno(f), offset) != 0) {
lua_pushstring(L, strerror(errno));
return 3;
if(fwrite(data, sizeof(char), len, f) == len) {
if(fflush(f) == 0) {
lua_pushboolean(L, 1); /* Great success! */
return 1;
} else {
err = errno;
}
return 2;
} else {
err = ferror(f);
}
fseek(f, offset, SEEK_SET);
/* Cut partially written data */
if(ftruncate(fileno(f), offset)) {
/* The file is now most likely corrupted, throw hard error */
return luaL_error(L, "atomic_append() failed in ftruncate(): %s", strerror(errno));
}
lua_pushnil(L);
lua_pushstring(L, strerror(err));
lua_pushinteger(L, err);
return 3;
}
/* Register functions */
@ -827,7 +831,7 @@ int luaopen_util_pposix(lua_State *L) {
{ "meminfo", lc_meminfo },
#endif
{ "fallocate", lc_fallocate },
{ "atomic_append", lc_atomic_append },
{ NULL, NULL }
};

View file

@ -17,7 +17,6 @@ local io_open = io.open;
local os_remove = os.remove;
local os_rename = os.rename;
local tonumber = tonumber;
local tostring = tostring;
local next = next;
local type = type;
local t_insert = table.insert;
@ -31,21 +30,12 @@ local path_separator = assert ( package.config:match ( "^([^\n]+)" ) , "package.
local prosody = prosody;
local raw_mkdir = lfs.mkdir;
local function fallocate(f, offset, len)
-- This assumes that current position == offset
local fake_data = (" "):rep(len);
local ok, msg = f:write(fake_data);
if not ok then
return ok, msg;
end
f:seek("set", offset);
return true;
end;
local atomic_append;
local ENOENT = 2;
pcall(function()
local pposix = require "util.pposix";
raw_mkdir = pposix.mkdir or raw_mkdir; -- Doesn't trample on umask
fallocate = pposix.fallocate or fallocate;
atomic_append = pposix.atomic_append;
ENOENT = pposix.ENOENT or ENOENT;
end);
@ -65,6 +55,19 @@ do
end
end
if not atomic_append then
function atomic_append(f, data)
local pos = f:seek();
if not f:write(data) or not f:flush() then
f:seek("set", pos);
f:write((" "):rep(#data));
f:flush();
return nil, "write-failed";
end
return true;
end
end
local _mkdir = {};
local function mkdir(path)
path = path:gsub("/", path_separator); -- TODO as an optimization, do this during path creation rather than here
@ -220,26 +223,16 @@ local function append(username, host, datastore, ext, data)
if type(data) ~= "string" then return; end
local filename = getpath(username, host, datastore, ext, true);
local ok;
local f, msg = io_open(filename, "r+");
local f = io_open(filename, "r+");
if not f then
return atomic_store(filename, data);
-- File did probably not exist, let's create it
end
local pos = f:seek("end");
ok, msg = fallocate(f, pos, #data);
if not ok then
log("warn", "fallocate() failed: %s", tostring(msg));
-- This doesn't work on every file system
end
if f:seek() ~= pos then
log("debug", "fallocate() changed file position");
f:seek("set", pos);
end
local ok, msg = atomic_append(f, data);
ok, msg = f:write(data);
if not ok then
f:close();
return ok, msg, "write";
@ -247,7 +240,7 @@ local function append(username, host, datastore, ext, data)
ok, msg = f:close();
if not ok then
return ok, msg;
return ok, msg, "close";
end
return true, pos;
@ -259,9 +252,10 @@ local function list_append(username, host, datastore, data)
-- save the datastore
data = "item(" .. serialize(data) .. ");\n";
local ok, msg = append(username, host, datastore, "list", data);
local ok, msg, where = append(username, host, datastore, "list", data);
if not ok then
log("error", "Unable to write to %s storage ('%s') for user: %s@%s", datastore, msg, username or "nil", host or "nil");
log("error", "Unable to write to %s storage ('%s' in %s) for user: %s@%s",
datastore, msg, where, username or "nil", host or "nil");
return ok, msg;
end
return true;