Wrote tests for history merging, then made them pass.

This commit is contained in:
ridiculousfish 2012-04-16 20:26:50 -07:00
parent d8428463d8
commit 3c190374b4
2 changed files with 139 additions and 3 deletions

View file

@ -762,9 +762,26 @@ static void test_history_matches(history_search_t &search, size_t matches) {
assert(! search.go_forwards()); assert(! search.go_forwards());
} }
static bool history_contains(history_t *history, const wcstring &txt) {
bool result = false;
size_t i;
for (i=1; ; i++) {
history_item_t item = history->item_at_index(i);
if (item.empty())
break;
if (item.str() == txt) {
result = true;
break;
}
}
return result;
}
class history_tests_t { class history_tests_t {
public: public:
static void test_history(void); static void test_history(void);
static void test_history_merge(void);
}; };
static wcstring random_string(void) { static wcstring random_string(void) {
@ -836,6 +853,70 @@ void history_tests_t::test_history(void) {
assert(bef.creation_timestamp == aft.creation_timestamp); assert(bef.creation_timestamp == aft.creation_timestamp);
assert(bef.required_paths == aft.required_paths); assert(bef.required_paths == aft.required_paths);
} }
/* Clean up after our tests */
history.clear();
}
// wait until the next second
static void time_barrier(void) {
time_t start = time(NULL);
do {
usleep(1000);
} while (time(NULL) == start);
}
void history_tests_t::test_history_merge(void) {
// In a single fish process, only one history is allowed to exist with the given name
// But it's common to have multiple history instances with the same name active in different processes,
// e.g. when you have multiple shells open.
// We try to get that right and merge all their history together. Test that case.
say( L"Testing history merge");
const size_t count = 3;
const wcstring name = L"merge_test";
history_t *hists[count] = {new history_t(name), new history_t(name), new history_t(name)};
wcstring texts[count] = {L"History 1", L"History 2", L"History 3"};
/* Make sure history is clear */
for (size_t i=0; i < count; i++) {
hists[i]->clear();
}
/* Make sure we don't add an item in the same second as we created the history */
time_barrier();
/* Add a different item to each */
for (size_t i=0; i < count; i++) {
hists[i]->add(texts[i]);
}
/* Save them */
for (size_t i=0; i < count; i++) {
hists[i]->save();
}
/* Make sure each history contains what it ought to, but they have not leaked into each other */
for (size_t i = 0; i < count; i++) {
for (size_t j=0; j < count; j++) {
bool does_contain = history_contains(hists[i], texts[j]);
bool should_contain = (i == j);
assert(should_contain == does_contain);
}
}
/* Make a new history. It should contain everything. The time_barrier() is so that the timestamp is newer, since we only pick up items whose timestamp is before the birth stamp. */
time_barrier();
history_t *everything = new history_t(name);
for (size_t i=0; i < count; i++) {
assert(history_contains(everything, texts[i]));
}
/* Clean up */
for (size_t i=0; i < 3; i++) {
delete hists[i];
}
everything->clear();
delete everything; //not as scary as it looks
} }
@ -873,6 +954,7 @@ int main( int argc, char **argv )
test_colors(); test_colors();
test_autosuggest(); test_autosuggest();
history_tests_t::test_history(); history_tests_t::test_history();
history_tests_t::test_history_merge();
say( L"Encountered %d errors in low-level tests", err_count ); say( L"Encountered %d errors in low-level tests", err_count );

View file

@ -240,6 +240,37 @@ static bool parse_timestamp(const char *str, time_t *out_when) {
return false; return false;
} }
// Returns a pointer to the start of the next line, or NULL
// The next line must itself end with a newline
// Note that the string is not null terminated
static const char *next_line(const char *start, size_t length) {
/* Handle the hopeless case */
if (length < 1)
return NULL;
/* Get a pointer to the end, that we must not pass */
const char * const end = start + length;
/* Skip past the next newline */
const char *nextline = (const char *)memchr(start, '\n', length);
if (! nextline || nextline >= end) {
return NULL;
}
/* Skip past the newline character itself */
if (++nextline >= end) {
return NULL;
}
/* Make sure this new line is itself "newline terminated". If it's not, return NULL; */
const char *next_newline = (const char *)memchr(nextline, '\n', end - nextline);
if (! next_newline) {
return NULL;
}
/* Done */
return nextline;
}
// Support for iteratively locating the offsets of history items // Support for iteratively locating the offsets of history items
// Pass the address and length of a mapped region. // Pass the address and length of a mapped region.
// Pass a pointer to a cursor size_t, initially 0 // Pass a pointer to a cursor size_t, initially 0
@ -251,7 +282,8 @@ static size_t offset_of_next_item(const char *begin, size_t mmap_length, size_t
size_t result = (size_t)(-1); size_t result = (size_t)(-1);
while (cursor < mmap_length) { while (cursor < mmap_length) {
const char * const line_start = begin + cursor; const char * const line_start = begin + cursor;
/* Look for a newline */
/* Advance the cursor to the next line */
const char *newline = (const char *)memchr(line_start, '\n', mmap_length - cursor); const char *newline = (const char *)memchr(line_start, '\n', mmap_length - cursor);
if (newline == NULL) if (newline == NULL)
break; break;
@ -274,14 +306,36 @@ static size_t offset_of_next_item(const char *begin, size_t mmap_length, size_t
! memcmp(line_start, "...", 3)) ! memcmp(line_start, "...", 3))
continue; continue;
/* A 0 timestamp means no cutoff */ /* At this point, we know line_start is at the beginning of an item. But maybe we want to skip this item because of timestamps. A 0 cutoff means we don't care; if we do care, then try parsing out a timestamp. */
if (cutoff_timestamp != 0) { if (cutoff_timestamp != 0) {
/* Hackish fast way to skip items created after our timestamp. This is the mechanism by which we avoid "seeing" commands from other sessions that started after we started. We try hard to ensure that our items are sorted by their timestamps, so in theory we could just break, but I don't think that works well if (for example) the clock changes. So we'll read all subsequent items. /* Hackish fast way to skip items created after our timestamp. This is the mechanism by which we avoid "seeing" commands from other sessions that started after we started. We try hard to ensure that our items are sorted by their timestamps, so in theory we could just break, but I don't think that works well if (for example) the clock changes. So we'll read all subsequent items.
*/ */
const char * const end = begin + mmap_length;
/* Walk over lines that we think are interior. These lines are not null terminated, but are guaranteed to contain a newline. */
bool has_timestamp = false;
time_t timestamp; time_t timestamp;
if (parse_timestamp(line_start, &timestamp) && timestamp >= cutoff_timestamp) const char *interior_line;
for (interior_line = next_line(line_start, end - line_start);
interior_line != NULL && ! has_timestamp;
interior_line = next_line(interior_line, end - interior_line)) {
/* If the first character is not a space, it's not an interior line, so we're done */
if (interior_line[0] != ' ')
break;
/* Hackish optimization: since we just stepped over some interior line, update the cursor so we don't have to look at these lines next time */
cursor = interior_line - begin;
/* Try parsing a timestamp from this line. If we succeed, the loop will break. */
has_timestamp = parse_timestamp(interior_line, &timestamp);
}
/* Skip this item if the timestamp is at or after our cutoff. */
if (has_timestamp && timestamp >= cutoff_timestamp) {
continue; continue;
} }
}
/* We made it through the gauntlet. */ /* We made it through the gauntlet. */
result = line_start - begin; result = line_start - begin;