Refactor parse_command
Patch status: merged
Patch by Tony Crisci
Long description:
parse_command returns a struct that contains useful information about the result of a command as a whole (instead of the intermediate representation used during parsing). parse_command now requires the caller to allocate the yajl_gen used for generating a json reply. This is passed as the second parameter to parse_command. If NULL is passed, no json reply will be generated.
To apply this patch, use:
curl http://cr.i3wm.org/patch/560/raw.patch | git am
b/include/commands_parser.h
| 25 |
@@ -17,7 +17,7 @@ |
| 26 |
* internally use this struct when calling cmd_floating and cmd_border. |
| 27 |
*/ |
| 28 |
struct CommandResultIR {
|
| 29 |
- /* The JSON generator to append a reply to. */ |
| 30 |
+ /* The JSON generator to append a reply to (may be NULL). */ |
| 31 |
yajl_gen json_gen; |
| 32 |
|
| 33 |
/* The next state to transition to. Passed to the function so that we can |
| 34 |
@@ -29,4 +29,31 @@ struct CommandResultIR {
|
| 35 |
bool needs_tree_render; |
| 36 |
}; |
| 37 |
|
| 38 |
-struct CommandResultIR *parse_command(const char *input); |
| 39 |
+typedef struct CommandResult CommandResult; |
| 40 |
+ |
| 41 |
+/** |
| 42 |
+ * A struct that contains useful information about the result of a command as a |
| 43 |
+ * whole (e.g. a compound command like "floating enable, border none"). |
| 44 |
+ * needs_tree_render is true if needs_tree_render of any individual command was |
| 45 |
+ * true. |
| 46 |
+ */ |
| 47 |
+struct CommandResult {
|
| 48 |
+ bool parse_error; |
| 49 |
+ /* the error_message is currently only set for parse errors */ |
| 50 |
+ char *error_message; |
| 51 |
+ bool needs_tree_render; |
| 52 |
+}; |
| 53 |
+ |
| 54 |
+/** |
| 55 |
+ * Parses and executes the given command. If a caller-allocated yajl_gen is |
| 56 |
+ * passed, a json reply will be generated in the format specified by the ipc |
| 57 |
+ * protocol. Pass NULL if no json reply is required. |
| 58 |
+ * |
| 59 |
+ * Free the returned CommandResult with command_result_free(). |
| 60 |
+ */ |
| 61 |
+CommandResult *parse_command(const char *input, yajl_gen gen); |
| 62 |
+ |
| 63 |
+/** |
| 64 |
+ * Frees a CommandResult |
| 65 |
+ */ |
| 66 |
+void command_result_free(CommandResult *result); |
b/src/assignments.c
| 71 |
@@ -45,13 +45,13 @@ void run_assignments(i3Window *window) {
|
| 72 |
DLOG("execute command %s\n", current->dest.command);
|
| 73 |
char *full_command; |
| 74 |
sasprintf(&full_command, "[id=\"%d\"] %s", window->id, current->dest.command); |
| 75 |
- struct CommandResultIR *command_output = parse_command(full_command); |
| 76 |
+ CommandResult *result = parse_command(full_command, NULL); |
| 77 |
free(full_command); |
| 78 |
|
| 79 |
- if (command_output->needs_tree_render) |
| 80 |
+ if (result->needs_tree_render) |
| 81 |
needs_tree_render = true; |
| 82 |
|
| 83 |
- yajl_gen_free(command_output->json_gen); |
| 84 |
+ command_result_free(result); |
| 85 |
} |
| 86 |
|
| 87 |
/* Store that we ran this assignment to not execute it again */ |
b/src/commands.c
| 92 |
@@ -16,21 +16,25 @@ |
| 93 |
#include "shmlog.h" |
| 94 |
|
| 95 |
// Macros to make the YAJL API a bit easier to use. |
| 96 |
-#define y(x, ...) yajl_gen_ ## x (cmd_output->json_gen, ##__VA_ARGS__) |
| 97 |
-#define ystr(str) yajl_gen_string(cmd_output->json_gen, (unsigned char*)str, strlen(str)) |
| 98 |
+#define y(x, ...) (cmd_output->json_gen != NULL ? yajl_gen_ ## x (cmd_output->json_gen, ##__VA_ARGS__) : 0) |
| 99 |
+#define ystr(str) (cmd_output->json_gen != NULL ? yajl_gen_string(cmd_output->json_gen, (unsigned char*)str, strlen(str)) : 0) |
| 100 |
#define ysuccess(success) do { \
|
| 101 |
- y(map_open); \ |
| 102 |
- ystr("success"); \
|
| 103 |
- y(bool, success); \ |
| 104 |
- y(map_close); \ |
| 105 |
+ if (cmd_output->json_gen != NULL) { \
|
| 106 |
+ y(map_open); \ |
| 107 |
+ ystr("success"); \
|
| 108 |
+ y(bool, success); \ |
| 109 |
+ y(map_close); \ |
| 110 |
+ } \ |
| 111 |
} while (0) |
| 112 |
#define yerror(message) do { \
|
| 113 |
- y(map_open); \ |
| 114 |
- ystr("success"); \
|
| 115 |
- y(bool, false); \ |
| 116 |
- ystr("error"); \
|
| 117 |
- ystr(message); \ |
| 118 |
- y(map_close); \ |
| 119 |
+ if (cmd_output->json_gen != NULL) { \
|
| 120 |
+ y(map_open); \ |
| 121 |
+ ystr("success"); \
|
| 122 |
+ y(bool, false); \ |
| 123 |
+ ystr("error"); \
|
| 124 |
+ ystr(message); \ |
| 125 |
+ y(map_close); \ |
| 126 |
+ } \ |
| 127 |
} while (0) |
| 128 |
|
| 129 |
/** When the command did not include match criteria (!), we use the currently |
b/src/commands_parser.c
| 134 |
@@ -35,8 +35,8 @@ |
| 135 |
#include "all.h" |
| 136 |
|
| 137 |
// Macros to make the YAJL API a bit easier to use. |
| 138 |
-#define y(x, ...) yajl_gen_ ## x (command_output.json_gen, ##__VA_ARGS__) |
| 139 |
-#define ystr(str) yajl_gen_string(command_output.json_gen, (unsigned char*)str, strlen(str)) |
| 140 |
+#define y(x, ...) (command_output.json_gen != NULL ? yajl_gen_ ## x (command_output.json_gen, ##__VA_ARGS__) : 0) |
| 141 |
+#define ystr(str) (command_output.json_gen != NULL ? yajl_gen_string(command_output.json_gen, (unsigned char*)str, strlen(str)) : 0) |
| 142 |
|
| 143 |
/******************************************************************************* |
| 144 |
* The data structures used for parsing. Essentially the current state and a |
| 145 |
@@ -205,12 +205,20 @@ static void next_state(const cmdp_token *token) {
|
| 146 |
} |
| 147 |
} |
| 148 |
|
| 149 |
-struct CommandResultIR *parse_command(const char *input) {
|
| 150 |
+/* |
| 151 |
+ * Parses and executes the given command. If a caller-allocated yajl_gen is |
| 152 |
+ * passed, a json reply will be generated in the format specified by the ipc |
| 153 |
+ * protocol. Pass NULL if no json reply is required. |
| 154 |
+ * |
| 155 |
+ * Free the returned CommandResult with command_result_free(). |
| 156 |
+ */ |
| 157 |
+CommandResult *parse_command(const char *input, yajl_gen gen) {
|
| 158 |
DLOG("COMMAND: *%s*\n", input);
|
| 159 |
state = INITIAL; |
| 160 |
+ CommandResult *result = scalloc(sizeof(CommandResult)); |
| 161 |
|
| 162 |
/* A YAJL JSON generator used for formatting replies. */ |
| 163 |
- command_output.json_gen = yajl_gen_alloc(NULL); |
| 164 |
+ command_output.json_gen = (gen != NULL ? gen : NULL); |
| 165 |
|
| 166 |
y(array_open); |
| 167 |
command_output.needs_tree_render = false; |
| 168 |
@@ -378,6 +386,9 @@ struct CommandResultIR *parse_command(const char *input) {
|
| 169 |
ELOG("Your command: %s\n", input);
|
| 170 |
ELOG(" %s\n", position);
|
| 171 |
|
| 172 |
+ result->parse_error = true; |
| 173 |
+ result->error_message = errormessage; |
| 174 |
+ |
| 175 |
/* Format this error message as a JSON reply. */ |
| 176 |
y(map_open); |
| 177 |
ystr("success");
|
| 178 |
@@ -396,7 +407,6 @@ struct CommandResultIR *parse_command(const char *input) {
|
| 179 |
y(map_close); |
| 180 |
|
| 181 |
free(position); |
| 182 |
- free(errormessage); |
| 183 |
clear_stack(); |
| 184 |
break; |
| 185 |
} |
| 186 |
@@ -404,7 +414,19 @@ struct CommandResultIR *parse_command(const char *input) {
|
| 187 |
|
| 188 |
y(array_close); |
| 189 |
|
| 190 |
- return &command_output; |
| 191 |
+ result->needs_tree_render = command_output.needs_tree_render; |
| 192 |
+ return result; |
| 193 |
+} |
| 194 |
+ |
| 195 |
+/* |
| 196 |
+ * Frees a CommandResult |
| 197 |
+ */ |
| 198 |
+void command_result_free(CommandResult *result) {
|
| 199 |
+ if (result == NULL) |
| 200 |
+ return; |
| 201 |
+ |
| 202 |
+ FREE(result->error_message); |
| 203 |
+ FREE(result); |
| 204 |
} |
| 205 |
|
| 206 |
/******************************************************************************* |
| 207 |
@@ -442,6 +464,12 @@ int main(int argc, char *argv[]) {
|
| 208 |
fprintf(stderr, "Syntax: %s <command>\n", argv[0]); |
| 209 |
return 1; |
| 210 |
} |
| 211 |
- parse_command(argv[1]); |
| 212 |
+ yajl_gen gen = yajl_gen_alloc(NULL); |
| 213 |
+ |
| 214 |
+ CommandResult *result = parse_command(argv[1], gen); |
| 215 |
+ |
| 216 |
+ command_result_free(result); |
| 217 |
+ |
| 218 |
+ yajl_gen_free(gen); |
| 219 |
} |
| 220 |
#endif |
b/src/ipc.c
| 225 |
@@ -117,20 +117,24 @@ IPC_HANDLER(command) {
|
| 226 |
char *command = scalloc(message_size + 1); |
| 227 |
strncpy(command, (const char*)message, message_size); |
| 228 |
LOG("IPC: received: *%s*\n", command);
|
| 229 |
- struct CommandResultIR *command_output = parse_command((const char*)command); |
| 230 |
+ yajl_gen gen = yajl_gen_alloc(NULL); |
| 231 |
+ |
| 232 |
+ CommandResult *result = parse_command((const char*)command, gen); |
| 233 |
free(command); |
| 234 |
|
| 235 |
- if (command_output->needs_tree_render) |
| 236 |
+ if (result->needs_tree_render) |
| 237 |
tree_render(); |
| 238 |
|
| 239 |
+ command_result_free(result); |
| 240 |
+ |
| 241 |
const unsigned char *reply; |
| 242 |
ylength length; |
| 243 |
- yajl_gen_get_buf(command_output->json_gen, &reply, &length); |
| 244 |
+ yajl_gen_get_buf(gen, &reply, &length); |
| 245 |
|
| 246 |
ipc_send_message(fd, length, I3_IPC_REPLY_TYPE_COMMAND, |
| 247 |
(const uint8_t*)reply); |
| 248 |
|
| 249 |
- yajl_gen_free(command_output->json_gen); |
| 250 |
+ yajl_gen_free(gen); |
| 251 |
} |
| 252 |
|
| 253 |
static void dump_rect(yajl_gen gen, const char *name, Rect r) {
|
b/src/key_press.c
| 258 |
@@ -72,19 +72,23 @@ void handle_key_press(xcb_key_press_event_t *event) {
|
| 259 |
if (bind == NULL) |
| 260 |
return; |
| 261 |
|
| 262 |
+ yajl_gen gen = yajl_gen_alloc(NULL); |
| 263 |
+ |
| 264 |
char *command_copy = sstrdup(bind->command); |
| 265 |
- struct CommandResultIR *command_output = parse_command(command_copy); |
| 266 |
+ CommandResult *result = parse_command(command_copy, gen); |
| 267 |
free(command_copy); |
| 268 |
|
| 269 |
- if (command_output->needs_tree_render) |
| 270 |
+ if (result->needs_tree_render) |
| 271 |
tree_render(); |
| 272 |
|
| 273 |
+ command_result_free(result); |
| 274 |
+ |
| 275 |
/* We parse the JSON reply to figure out whether there was an error |
| 276 |
* ("success" being false in on of the returned dictionaries). */
|
| 277 |
const unsigned char *reply; |
| 278 |
size_t length; |
| 279 |
yajl_handle handle = yajl_alloc(&command_error_callbacks, NULL, NULL); |
| 280 |
- yajl_gen_get_buf(command_output->json_gen, &reply, &length); |
| 281 |
+ yajl_gen_get_buf(gen, &reply, &length); |
| 282 |
|
| 283 |
current_nesting_level = 0; |
| 284 |
parse_error_key = false; |
| 285 |
@@ -116,5 +120,5 @@ void handle_key_press(xcb_key_press_event_t *event) {
|
| 286 |
|
| 287 |
yajl_free(handle); |
| 288 |
|
| 289 |
- yajl_gen_free(command_output->json_gen); |
| 290 |
+ yajl_gen_free(gen); |
| 291 |
} |