mirror of
https://github.com/systemd/systemd.git
synced 2025-01-12 13:18:14 +03:00
journald: when we detect the journal file we are about to write to has been deleted, rotate
https://bugzilla.redhat.com/show_bug.cgi?id=1171719
This commit is contained in:
parent
aba843317d
commit
2678031a17
@ -67,6 +67,9 @@
|
|||||||
/* How much to increase the journal file size at once each time we allocate something new. */
|
/* How much to increase the journal file size at once each time we allocate something new. */
|
||||||
#define FILE_SIZE_INCREASE (8ULL*1024ULL*1024ULL) /* 8MB */
|
#define FILE_SIZE_INCREASE (8ULL*1024ULL*1024ULL) /* 8MB */
|
||||||
|
|
||||||
|
/* Reread fstat() of the file for detecting deletions at least this often */
|
||||||
|
#define LAST_STAT_REFRESH_USEC (5*USEC_PER_SEC)
|
||||||
|
|
||||||
/* The mmap context to use for the header we pick as one above the last defined typed */
|
/* The mmap context to use for the header we pick as one above the last defined typed */
|
||||||
#define CONTEXT_HEADER _OBJECT_TYPE_MAX
|
#define CONTEXT_HEADER _OBJECT_TYPE_MAX
|
||||||
|
|
||||||
@ -319,6 +322,22 @@ static int journal_file_verify_header(JournalFile *f) {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int journal_file_fstat(JournalFile *f) {
|
||||||
|
assert(f);
|
||||||
|
assert(f->fd >= 0);
|
||||||
|
|
||||||
|
if (fstat(f->fd, &f->last_stat) < 0)
|
||||||
|
return -errno;
|
||||||
|
|
||||||
|
f->last_stat_usec = now(CLOCK_MONOTONIC);
|
||||||
|
|
||||||
|
/* Refuse appending to files that are already deleted */
|
||||||
|
if (f->last_stat.st_nlink <= 0)
|
||||||
|
return -EIDRM;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static int journal_file_allocate(JournalFile *f, uint64_t offset, uint64_t size) {
|
static int journal_file_allocate(JournalFile *f, uint64_t offset, uint64_t size) {
|
||||||
uint64_t old_size, new_size;
|
uint64_t old_size, new_size;
|
||||||
int r;
|
int r;
|
||||||
@ -340,8 +359,21 @@ static int journal_file_allocate(JournalFile *f, uint64_t offset, uint64_t size)
|
|||||||
if (new_size < le64toh(f->header->header_size))
|
if (new_size < le64toh(f->header->header_size))
|
||||||
new_size = le64toh(f->header->header_size);
|
new_size = le64toh(f->header->header_size);
|
||||||
|
|
||||||
if (new_size <= old_size)
|
if (new_size <= old_size) {
|
||||||
return 0;
|
|
||||||
|
/* We already pre-allocated enough space, but before
|
||||||
|
* we write to it, let's check with fstat() if the
|
||||||
|
* file got deleted, in order make sure we don't throw
|
||||||
|
* away the data immediately. Don't check fstat() for
|
||||||
|
* all writes though, but only once ever 10s. */
|
||||||
|
|
||||||
|
if (f->last_stat_usec + LAST_STAT_REFRESH_USEC > now(CLOCK_MONOTONIC))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return journal_file_fstat(f);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Allocate more space. */
|
||||||
|
|
||||||
if (f->metrics.max_size > 0 && new_size > f->metrics.max_size)
|
if (f->metrics.max_size > 0 && new_size > f->metrics.max_size)
|
||||||
return -E2BIG;
|
return -E2BIG;
|
||||||
@ -376,12 +408,9 @@ static int journal_file_allocate(JournalFile *f, uint64_t offset, uint64_t size)
|
|||||||
if (r != 0)
|
if (r != 0)
|
||||||
return -r;
|
return -r;
|
||||||
|
|
||||||
if (fstat(f->fd, &f->last_stat) < 0)
|
|
||||||
return -errno;
|
|
||||||
|
|
||||||
f->header->arena_size = htole64(new_size - le64toh(f->header->header_size));
|
f->header->arena_size = htole64(new_size - le64toh(f->header->header_size));
|
||||||
|
|
||||||
return 0;
|
return journal_file_fstat(f);
|
||||||
}
|
}
|
||||||
|
|
||||||
static unsigned type_to_context(ObjectType type) {
|
static unsigned type_to_context(ObjectType type) {
|
||||||
@ -392,6 +421,8 @@ static unsigned type_to_context(ObjectType type) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int journal_file_move_to(JournalFile *f, ObjectType type, bool keep_always, uint64_t offset, uint64_t size, void **ret) {
|
static int journal_file_move_to(JournalFile *f, ObjectType type, bool keep_always, uint64_t offset, uint64_t size, void **ret) {
|
||||||
|
int r;
|
||||||
|
|
||||||
assert(f);
|
assert(f);
|
||||||
assert(ret);
|
assert(ret);
|
||||||
|
|
||||||
@ -403,8 +434,11 @@ static int journal_file_move_to(JournalFile *f, ObjectType type, bool keep_alway
|
|||||||
/* Hmm, out of range? Let's refresh the fstat() data
|
/* Hmm, out of range? Let's refresh the fstat() data
|
||||||
* first, before we trust that check. */
|
* first, before we trust that check. */
|
||||||
|
|
||||||
if (fstat(f->fd, &f->last_stat) < 0 ||
|
r = journal_file_fstat(f);
|
||||||
offset + size > (uint64_t) f->last_stat.st_size)
|
if (r < 0)
|
||||||
|
return r;
|
||||||
|
|
||||||
|
if (offset + size > (uint64_t) f->last_stat.st_size)
|
||||||
return -EADDRNOTAVAIL;
|
return -EADDRNOTAVAIL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2548,10 +2582,9 @@ int journal_file_open(
|
|||||||
goto fail;
|
goto fail;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (fstat(f->fd, &f->last_stat) < 0) {
|
r = journal_file_fstat(f);
|
||||||
r = -errno;
|
if (r < 0)
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
|
||||||
|
|
||||||
if (f->last_stat.st_size == 0 && f->writable) {
|
if (f->last_stat.st_size == 0 && f->writable) {
|
||||||
/* Let's attach the creation time to the journal file,
|
/* Let's attach the creation time to the journal file,
|
||||||
@ -2580,10 +2613,9 @@ int journal_file_open(
|
|||||||
if (r < 0)
|
if (r < 0)
|
||||||
goto fail;
|
goto fail;
|
||||||
|
|
||||||
if (fstat(f->fd, &f->last_stat) < 0) {
|
r = journal_file_fstat(f);
|
||||||
r = -errno;
|
if (r < 0)
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
|
||||||
|
|
||||||
newly_created = true;
|
newly_created = true;
|
||||||
}
|
}
|
||||||
@ -2700,8 +2732,11 @@ int journal_file_rotate(JournalFile **f, bool compress, bool seal) {
|
|||||||
if (r < 0)
|
if (r < 0)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
/* Try to rename the file to the archived version. If the file
|
||||||
|
* already was deleted, we'll get ENOENT, let's ignore that
|
||||||
|
* case. */
|
||||||
r = rename(old_file->path, p);
|
r = rename(old_file->path, p);
|
||||||
if (r < 0)
|
if (r < 0 && errno != ENOENT)
|
||||||
return -errno;
|
return -errno;
|
||||||
|
|
||||||
old_file->header->state = STATE_ARCHIVED;
|
old_file->header->state = STATE_ARCHIVED;
|
||||||
|
@ -82,6 +82,7 @@ typedef struct JournalFile {
|
|||||||
|
|
||||||
char *path;
|
char *path;
|
||||||
struct stat last_stat;
|
struct stat last_stat;
|
||||||
|
usec_t last_stat_usec;
|
||||||
|
|
||||||
Header *header;
|
Header *header;
|
||||||
HashItem *data_hash_table;
|
HashItem *data_hash_table;
|
||||||
|
@ -318,6 +318,7 @@ static int do_rotate(
|
|||||||
log_error_errno(r, "Failed to create new %s journal: %m", name);
|
log_error_errno(r, "Failed to create new %s journal: %m", name);
|
||||||
else
|
else
|
||||||
server_fix_perms(s, *f, uid);
|
server_fix_perms(s, *f, uid);
|
||||||
|
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -466,7 +467,8 @@ static bool shall_try_append_again(JournalFile *f, int r) {
|
|||||||
-EPROTONOSUPPORT Unsupported feature
|
-EPROTONOSUPPORT Unsupported feature
|
||||||
-EBADMSG Corrupted
|
-EBADMSG Corrupted
|
||||||
-ENODATA Truncated
|
-ENODATA Truncated
|
||||||
-ESHUTDOWN Already archived */
|
-ESHUTDOWN Already archived
|
||||||
|
-EIDRM Journal file has been deleted */
|
||||||
|
|
||||||
if (r == -E2BIG || r == -EFBIG || r == -EDQUOT || r == -ENOSPC)
|
if (r == -E2BIG || r == -EFBIG || r == -EDQUOT || r == -ENOSPC)
|
||||||
log_debug("%s: Allocation limit reached, rotating.", f->path);
|
log_debug("%s: Allocation limit reached, rotating.", f->path);
|
||||||
@ -480,6 +482,8 @@ static bool shall_try_append_again(JournalFile *f, int r) {
|
|||||||
log_warning("%s: Journal file corrupted, rotating.", f->path);
|
log_warning("%s: Journal file corrupted, rotating.", f->path);
|
||||||
else if (r == -EIO)
|
else if (r == -EIO)
|
||||||
log_warning("%s: IO error, rotating.", f->path);
|
log_warning("%s: IO error, rotating.", f->path);
|
||||||
|
else if (r == -EIDRM)
|
||||||
|
log_warning("%s: Journal file has been deleted, rotating.", f->path);
|
||||||
else
|
else
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
|
@ -39,8 +39,6 @@ int main(int argc, char *argv[]) {
|
|||||||
r = journal_file_open(fn, O_CREAT|O_RDWR, 0644, false, false, NULL, NULL, NULL, &new_journal);
|
r = journal_file_open(fn, O_CREAT|O_RDWR, 0644, false, false, NULL, NULL, NULL, &new_journal);
|
||||||
assert_se(r >= 0);
|
assert_se(r >= 0);
|
||||||
|
|
||||||
unlink(fn);
|
|
||||||
|
|
||||||
r = sd_journal_open(&j, 0);
|
r = sd_journal_open(&j, 0);
|
||||||
assert_se(r >= 0);
|
assert_se(r >= 0);
|
||||||
|
|
||||||
@ -68,6 +66,7 @@ int main(int argc, char *argv[]) {
|
|||||||
|
|
||||||
journal_file_close(new_journal);
|
journal_file_close(new_journal);
|
||||||
|
|
||||||
|
unlink(fn);
|
||||||
assert_se(rmdir(dn) == 0);
|
assert_se(rmdir(dn) == 0);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
Loading…
Reference in New Issue
Block a user