Bugfix: free incomplete containers when JSON parsing fails
related to #2755
This commit is contained in:
parent
d35de66f1e
commit
f120a9d929
|
@ -25,6 +25,12 @@ Con *con_new_skeleton(Con *parent, i3Window *window);
|
||||||
*/
|
*/
|
||||||
Con *con_new(Con *parent, i3Window *window);
|
Con *con_new(Con *parent, i3Window *window);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Frees the specified container.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
void con_free(Con *con);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets input focus to the given container. Will be updated in X11 in the next
|
* Sets input focus to the given container. Will be updated in X11 in the next
|
||||||
* run of x_push_changes().
|
* run of x_push_changes().
|
||||||
|
|
24
src/con.c
24
src/con.c
|
@ -73,6 +73,30 @@ Con *con_new(Con *parent, i3Window *window) {
|
||||||
return new;
|
return new;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Frees the specified container.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
void con_free(Con *con) {
|
||||||
|
free(con->name);
|
||||||
|
FREE(con->deco_render_params);
|
||||||
|
TAILQ_REMOVE(&all_cons, con, all_cons);
|
||||||
|
while (!TAILQ_EMPTY(&(con->swallow_head))) {
|
||||||
|
Match *match = TAILQ_FIRST(&(con->swallow_head));
|
||||||
|
TAILQ_REMOVE(&(con->swallow_head), match, matches);
|
||||||
|
match_free(match);
|
||||||
|
free(match);
|
||||||
|
}
|
||||||
|
while (!TAILQ_EMPTY(&(con->marks_head))) {
|
||||||
|
mark_t *mark = TAILQ_FIRST(&(con->marks_head));
|
||||||
|
TAILQ_REMOVE(&(con->marks_head), mark, marks);
|
||||||
|
FREE(mark->name);
|
||||||
|
FREE(mark);
|
||||||
|
}
|
||||||
|
free(con);
|
||||||
|
DLOG("con %p freed\n", con);
|
||||||
|
}
|
||||||
|
|
||||||
static void _con_attach(Con *con, Con *parent, Con *previous, bool ignore_focus) {
|
static void _con_attach(Con *con, Con *parent, Con *previous, bool ignore_focus) {
|
||||||
con->parent = parent;
|
con->parent = parent;
|
||||||
Con *loop;
|
Con *loop;
|
||||||
|
|
|
@ -18,6 +18,7 @@
|
||||||
/* TODO: refactor the whole parsing thing */
|
/* TODO: refactor the whole parsing thing */
|
||||||
|
|
||||||
static char *last_key;
|
static char *last_key;
|
||||||
|
static int incomplete;
|
||||||
static Con *json_node;
|
static Con *json_node;
|
||||||
static Con *to_focus;
|
static Con *to_focus;
|
||||||
static bool parsing_swallows;
|
static bool parsing_swallows;
|
||||||
|
@ -68,6 +69,9 @@ static int json_start_map(void *ctx) {
|
||||||
json_node->name = NULL;
|
json_node->name = NULL;
|
||||||
json_node->parent = parent;
|
json_node->parent = parent;
|
||||||
}
|
}
|
||||||
|
/* json_node is incomplete and should be removed if parsing fails */
|
||||||
|
incomplete++;
|
||||||
|
DLOG("incomplete = %d\n", incomplete);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return 1;
|
return 1;
|
||||||
|
@ -166,6 +170,8 @@ static int json_end_map(void *ctx) {
|
||||||
LOG("Creating window\n");
|
LOG("Creating window\n");
|
||||||
x_con_init(json_node);
|
x_con_init(json_node);
|
||||||
json_node = json_node->parent;
|
json_node = json_node->parent;
|
||||||
|
incomplete--;
|
||||||
|
DLOG("incomplete = %d\n", incomplete);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (parsing_swallows && swallow_is_empty) {
|
if (parsing_swallows && swallow_is_empty) {
|
||||||
|
@ -634,6 +640,7 @@ void tree_append_json(Con *con, const char *filename, char **errormsg) {
|
||||||
yajl_status stat;
|
yajl_status stat;
|
||||||
json_node = con;
|
json_node = con;
|
||||||
to_focus = NULL;
|
to_focus = NULL;
|
||||||
|
incomplete = 0;
|
||||||
parsing_swallows = false;
|
parsing_swallows = false;
|
||||||
parsing_rect = false;
|
parsing_rect = false;
|
||||||
parsing_deco_rect = false;
|
parsing_deco_rect = false;
|
||||||
|
@ -649,6 +656,12 @@ void tree_append_json(Con *con, const char *filename, char **errormsg) {
|
||||||
if (errormsg != NULL)
|
if (errormsg != NULL)
|
||||||
*errormsg = sstrdup((const char *)str);
|
*errormsg = sstrdup((const char *)str);
|
||||||
yajl_free_error(hand, str);
|
yajl_free_error(hand, str);
|
||||||
|
while (incomplete-- > 0) {
|
||||||
|
Con *parent = json_node->parent;
|
||||||
|
DLOG("freeing incomplete container %p\n", json_node);
|
||||||
|
con_free(json_node);
|
||||||
|
json_node = parent;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* In case not all containers were restored, we need to fix the
|
/* In case not all containers were restored, we need to fix the
|
||||||
|
|
17
src/tree.c
17
src/tree.c
|
@ -320,22 +320,7 @@ bool tree_close_internal(Con *con, kill_window_t kill_window, bool dont_kill_par
|
||||||
DLOG("parent container killed\n");
|
DLOG("parent container killed\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
free(con->name);
|
con_free(con);
|
||||||
FREE(con->deco_render_params);
|
|
||||||
TAILQ_REMOVE(&all_cons, con, all_cons);
|
|
||||||
while (!TAILQ_EMPTY(&(con->swallow_head))) {
|
|
||||||
Match *match = TAILQ_FIRST(&(con->swallow_head));
|
|
||||||
TAILQ_REMOVE(&(con->swallow_head), match, matches);
|
|
||||||
match_free(match);
|
|
||||||
free(match);
|
|
||||||
}
|
|
||||||
while (!TAILQ_EMPTY(&(con->marks_head))) {
|
|
||||||
mark_t *mark = TAILQ_FIRST(&(con->marks_head));
|
|
||||||
TAILQ_REMOVE(&(con->marks_head), mark, marks);
|
|
||||||
FREE(mark->name);
|
|
||||||
FREE(mark);
|
|
||||||
}
|
|
||||||
free(con);
|
|
||||||
|
|
||||||
/* in the case of floating windows, we already focused another container
|
/* in the case of floating windows, we already focused another container
|
||||||
* when closing the parent, so we can exit now. */
|
* when closing the parent, so we can exit now. */
|
||||||
|
|
|
@ -131,14 +131,103 @@ print $fh <<'EOT';
|
||||||
EOT
|
EOT
|
||||||
$fh->flush;
|
$fh->flush;
|
||||||
my $reply = cmd "append_layout $filename";
|
my $reply = cmd "append_layout $filename";
|
||||||
diag('reply = ' . Dumper($reply));
|
ok(!$reply->[0]->{success}, 'IPC reply did not indicate success');
|
||||||
|
|
||||||
does_i3_live;
|
does_i3_live;
|
||||||
|
|
||||||
ok(!$reply->[0]->{success}, 'IPC reply did not indicate success');
|
|
||||||
|
|
||||||
close($fh);
|
close($fh);
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# another file with a superfluous trailing comma (issue #2755)
|
||||||
|
################################################################################
|
||||||
|
|
||||||
|
subtest 'issue 2755' => sub {
|
||||||
|
plan tests => 4;
|
||||||
|
$ws = fresh_workspace;
|
||||||
|
|
||||||
|
@content = @{get_ws_content($ws)};
|
||||||
|
is(@content, 0, 'no nodes on the new workspace yet');
|
||||||
|
|
||||||
|
($fh, $filename) = tempfile(UNLINK => 1);
|
||||||
|
print $fh <<'EOT';
|
||||||
|
// vim:ts=4:sw=4:et
|
||||||
|
{
|
||||||
|
// splith split container with 2 children
|
||||||
|
"border": "normal",
|
||||||
|
"floating": "auto_off",
|
||||||
|
"layout": "splith",
|
||||||
|
"percent": null,
|
||||||
|
"type": "con",
|
||||||
|
"nodes": [
|
||||||
|
{
|
||||||
|
"border": "normal",
|
||||||
|
"current_border_width": 2,
|
||||||
|
"floating": "auto_off",
|
||||||
|
"geometry": {
|
||||||
|
"height": 860,
|
||||||
|
"width": 1396,
|
||||||
|
"x": 1922,
|
||||||
|
"y": 38
|
||||||
|
},
|
||||||
|
"name": "Chromium1",
|
||||||
|
"percent": 0.5,
|
||||||
|
"swallows": [
|
||||||
|
{
|
||||||
|
"class": "^Chromium$",
|
||||||
|
// "instance": "^chromium$",
|
||||||
|
// "title": "^Git\\ Tutorial\\ \\-\\ corp\\ \\-\\ Chromium$",
|
||||||
|
// "transient_for": "^$",
|
||||||
|
// "window_role": "^browser$"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"type": "con"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"border": "normal",
|
||||||
|
"current_border_width": 2,
|
||||||
|
"floating": "auto_off",
|
||||||
|
"geometry": {
|
||||||
|
"height": 1040,
|
||||||
|
"width": 956,
|
||||||
|
"x": 2,
|
||||||
|
"y": 38
|
||||||
|
},
|
||||||
|
"name": "Chromium2",
|
||||||
|
"percent": 0.5,
|
||||||
|
"swallows": [
|
||||||
|
{
|
||||||
|
"class": "^Chromium$",
|
||||||
|
// "instance": "^chromium$",
|
||||||
|
// "title": "^Nutanix\\ \\-\\ Prod\\ \\-\\ Sign\\ In\\ \\-\\ Chromium$",
|
||||||
|
// "transient_for": "^$",
|
||||||
|
// "window_role": "^browser$"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"type": "con"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
EOT
|
||||||
|
$fh->flush;
|
||||||
|
$reply = cmd "append_layout $filename";
|
||||||
|
ok(!$reply->[0]->{success}, 'IPC reply indicated success');
|
||||||
|
|
||||||
|
does_i3_live;
|
||||||
|
|
||||||
|
# Move to a different workspace rendered the half-attached con’s con->parent
|
||||||
|
# invalid.
|
||||||
|
fresh_workspace;
|
||||||
|
|
||||||
|
cmd '[urgent=latest] focus';
|
||||||
|
$reply = cmd 'scratchpad show';
|
||||||
|
|
||||||
|
does_i3_live;
|
||||||
|
|
||||||
|
close($fh);
|
||||||
|
};
|
||||||
|
|
||||||
################################################################################
|
################################################################################
|
||||||
# wrong percent key in a child node
|
# wrong percent key in a child node
|
||||||
################################################################################
|
################################################################################
|
||||||
|
|
Loading…
Reference in New Issue