470 lines
10 KiB
C
470 lines
10 KiB
C
#include "wmcliphist.h"
|
|
#include <sys/stat.h>
|
|
|
|
#define RC_BUF_SIZE 256
|
|
|
|
/* automat state */
|
|
typedef enum {
|
|
STATE_BEGINING,
|
|
STATE_COMMENT,
|
|
STATE_DIRECTIVE,
|
|
STATE_VALUE,
|
|
STATE_EXPRESSION_START,
|
|
STATE_EXPRESSION,
|
|
STATE_EXPRESSION_END,
|
|
STATE_ACTION,
|
|
STATE_COMMAND
|
|
} STATE;
|
|
|
|
|
|
GList *action_list = NULL;
|
|
|
|
|
|
/* add character to buffer */
|
|
#define add_to_buff(buf, pos, chr) do { \
|
|
buf[pos++] = chr; \
|
|
buf[pos] = '\0'; \
|
|
if (pos + 1 == RC_BUF_SIZE) \
|
|
error = 7; \
|
|
} while (0)
|
|
|
|
|
|
#define ismywhite(c) (c == '\t' || c == ' ')
|
|
|
|
|
|
/*
|
|
* returns config/data file name in user's home
|
|
*/
|
|
char *
|
|
rcconfig_get_name(char *append)
|
|
{
|
|
static gchar fname[PATH_MAX];
|
|
const gchar *home;
|
|
|
|
begin_func("rcconfig_get_name");
|
|
|
|
if (!(home = g_getenv("HOME")))
|
|
return_val(NULL);
|
|
|
|
memset(fname, 0, PATH_MAX);
|
|
snprintf(fname, PATH_MAX, "%s/.wmcliphist%s", home, append);
|
|
|
|
return_val(fname);
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
* appends parsed action to action list
|
|
*/
|
|
int
|
|
action_append(char *expr_buf, char *action_buf, char *cmd_buf)
|
|
{
|
|
ACTION *action;
|
|
|
|
begin_func("action_append");
|
|
|
|
action = g_new0(ACTION, 1);
|
|
|
|
if (regcomp(&action->expression, expr_buf,
|
|
REG_EXTENDED|REG_ICASE|REG_NOSUB) != 0) {
|
|
g_free(action);
|
|
return_val(-101);
|
|
}
|
|
|
|
if (strcmp(action_buf, "exec") == 0)
|
|
action->action = ACT_EXEC;
|
|
else if (strcmp(action_buf, "submenu") == 0)
|
|
action->action = ACT_SUBMENU;
|
|
else if (strcmp(action_buf, "ignore") == 0)
|
|
action->action = ACT_IGNORE;
|
|
else {
|
|
g_free(action);
|
|
return_val(-102);
|
|
}
|
|
|
|
action->command = g_strdup(cmd_buf);
|
|
|
|
action_list = g_list_append(action_list, action);
|
|
|
|
return_val(0);
|
|
}
|
|
|
|
|
|
/*
|
|
* read and parse rcconfig
|
|
*/
|
|
int
|
|
rcconfig_get(char *fname)
|
|
{
|
|
int f_rc;
|
|
char tmp[1024], c;
|
|
int byte_cnt;
|
|
STATE state = STATE_BEGINING;
|
|
char direc_buf[RC_BUF_SIZE],
|
|
expr_buf[RC_BUF_SIZE],
|
|
action_buf[RC_BUF_SIZE],
|
|
cmd_buf[RC_BUF_SIZE];
|
|
int buf_index = 0;
|
|
int error = 0, eof = 0;
|
|
int i;
|
|
int line = 0;
|
|
int res;
|
|
|
|
begin_func("rcconfig_get");
|
|
|
|
close(open(fname, O_CREAT, S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH));
|
|
|
|
if ((f_rc = open(fname, O_RDONLY)) < 0) {
|
|
fprintf(stderr, ".wmcliphistrc not found\n");
|
|
return_val(1);
|
|
}
|
|
|
|
i = byte_cnt = 0;
|
|
while (1) {
|
|
if (i == byte_cnt) {
|
|
byte_cnt = read(f_rc, tmp, 1024);
|
|
if (byte_cnt == -1) {
|
|
fprintf(stderr, "cannot read .wmcliphistrc\n");
|
|
break;
|
|
} else if (byte_cnt < 1024) {
|
|
tmp[byte_cnt++] = 0;
|
|
eof = 1;
|
|
}
|
|
i = 0;
|
|
}
|
|
|
|
c = tmp[i++];
|
|
switch (state) {
|
|
case STATE_BEGINING:
|
|
line++;
|
|
if (isalnum(c)) {
|
|
state = STATE_DIRECTIVE;
|
|
buf_index = 0;
|
|
add_to_buff(direc_buf, buf_index, c);
|
|
} else if (c == '#')
|
|
state = STATE_COMMENT;
|
|
else if (ismywhite(c))
|
|
line--;
|
|
else if (c == '\n')
|
|
state = STATE_BEGINING;
|
|
else
|
|
error = 1;
|
|
break;
|
|
|
|
case STATE_COMMENT:
|
|
if (c == '\n' || c == '\0')
|
|
state = STATE_BEGINING;
|
|
break;
|
|
|
|
case STATE_DIRECTIVE:
|
|
if (ismywhite(c)) {
|
|
if (strcmp(direc_buf, "action") == 0) {
|
|
state = STATE_EXPRESSION_START;
|
|
buf_index = 0;
|
|
} else if (strcmp(direc_buf, "menukey") == 0) {
|
|
state = STATE_VALUE;
|
|
buf_index = 0;
|
|
} else if (strcmp(direc_buf, "prev_item_key") == 0) {
|
|
state = STATE_VALUE;
|
|
buf_index = 0;
|
|
} else if (strcmp(direc_buf, "exec_item_key") == 0) {
|
|
state = STATE_VALUE;
|
|
buf_index = 0;
|
|
} else if (strcmp(direc_buf, "keep") == 0) {
|
|
state = STATE_VALUE;
|
|
buf_index = 0;
|
|
} else if (strcmp(direc_buf, "lcolor") == 0) {
|
|
state = STATE_VALUE;
|
|
buf_index = 0;
|
|
} else if (strcmp(direc_buf, "clipboard") == 0) {
|
|
state = STATE_VALUE;
|
|
buf_index = 0;
|
|
} else if (strcmp(direc_buf, "autosave") == 0) {
|
|
state = STATE_VALUE;
|
|
buf_index = 0;
|
|
} else if (strcmp(direc_buf,
|
|
"confirm_exec") == 0) {
|
|
state = STATE_VALUE;
|
|
buf_index = 0;
|
|
} else if (strcmp(direc_buf, "exec_immediately") == 0) {
|
|
state = STATE_VALUE;
|
|
buf_index = 0;
|
|
} else if (strcmp(direc_buf, "exec_middleclick") == 0) {
|
|
state = STATE_VALUE;
|
|
buf_index = 0;
|
|
} else if (strcmp(direc_buf, "exec_hotkey") == 0) {
|
|
state = STATE_VALUE;
|
|
buf_index = 0;
|
|
} else if (strcmp(direc_buf, "auto_take_up") == 0) {
|
|
state = STATE_VALUE;
|
|
buf_index = 0;
|
|
} else {
|
|
error = 8;
|
|
}
|
|
} else if (isalpha(c) || c == '_') {
|
|
add_to_buff(direc_buf, buf_index, c);
|
|
} else {
|
|
error = 1;
|
|
}
|
|
break;
|
|
|
|
case STATE_VALUE:
|
|
if (c == '\n' || ismywhite(c)) {
|
|
if (strcmp(direc_buf, "menukey") == 0) {
|
|
memset(menukey_str, 0, 32);
|
|
strncpy(menukey_str,
|
|
expr_buf, 31);
|
|
} else if (strcmp(direc_buf, "prev_item_key") == 0) {
|
|
memset(prev_item_key_str, 0, 32);
|
|
strncpy(prev_item_key_str,
|
|
expr_buf, 31);
|
|
} else if (strcmp(direc_buf, "exec_item_key") == 0) {
|
|
memset(exec_item_key_str, 0, 32);
|
|
strncpy(exec_item_key_str,
|
|
expr_buf, 31);
|
|
} else if (strcmp(direc_buf, "keep") == 0) {
|
|
num_items_to_keep =
|
|
atoi(expr_buf);
|
|
} else if (strcmp(direc_buf, "lcolor")
|
|
== 0) {
|
|
memset(locked_color_str, 0, 32);
|
|
strncpy(locked_color_str,
|
|
expr_buf, 31);
|
|
} else if (strcmp(direc_buf, "clipboard")
|
|
== 0) {
|
|
memset(clipboard_str, 0, 32);
|
|
strncpy(clipboard_str,
|
|
expr_buf, 31);
|
|
} else if (strcmp(direc_buf, "autosave") == 0) {
|
|
autosave_period =
|
|
atoi(expr_buf);
|
|
} else if (strcmp(direc_buf,
|
|
"confirm_exec") == 0) {
|
|
if (strcasecmp(expr_buf, "yes") == 0) {
|
|
confirm_exec = 1;
|
|
}
|
|
} else if (strcmp(direc_buf, "exec_immediately") == 0) {
|
|
if (strcasecmp(expr_buf, "no") == 0) {
|
|
exec_immediately= 0;
|
|
}
|
|
} else if (strcmp(direc_buf, "exec_middleclick") == 0) {
|
|
if (strcasecmp(expr_buf, "no") == 0) {
|
|
exec_middleclick = 0;
|
|
}
|
|
} else if (strcmp(direc_buf, "exec_hotkey") == 0) {
|
|
if (strcasecmp(expr_buf, "no") == 0) {
|
|
exec_hotkey = 0;
|
|
}
|
|
} else if (strcmp(direc_buf, "auto_take_up") == 0) {
|
|
if (strcasecmp(expr_buf, "no") == 0) {
|
|
auto_take_up = 0;
|
|
}
|
|
} else
|
|
error = 1;
|
|
} else if (isgraph(c))
|
|
add_to_buff(expr_buf, buf_index, c);
|
|
else
|
|
error = 2;
|
|
|
|
if (error == 0) {
|
|
if (c == '\n' || c == '\0') {
|
|
buf_index = 0;
|
|
state = STATE_BEGINING;
|
|
} else if (ismywhite(c)) {
|
|
buf_index = 0;
|
|
state = STATE_COMMENT;
|
|
}
|
|
}
|
|
|
|
break;
|
|
|
|
case STATE_EXPRESSION_START:
|
|
if (c == '"')
|
|
state = STATE_EXPRESSION;
|
|
else
|
|
error = 1;
|
|
break;
|
|
|
|
case STATE_EXPRESSION:
|
|
if (c == '"')
|
|
state = STATE_EXPRESSION_END;
|
|
else if (c == '\n')
|
|
error = 1;
|
|
else
|
|
add_to_buff(expr_buf, buf_index, c);
|
|
break;
|
|
|
|
case STATE_EXPRESSION_END:
|
|
if (c != ' ' && c != '\t')
|
|
error = 1;
|
|
if (strcmp(direc_buf, "action") == 0) {
|
|
state = STATE_ACTION;
|
|
buf_index = 0;
|
|
} else
|
|
error = 1;
|
|
|
|
break;
|
|
|
|
case STATE_ACTION:
|
|
if (c == ' ' || c == '\t') {
|
|
state = STATE_COMMAND;
|
|
buf_index = 0;
|
|
} else if (c == '\0' || c == '\n') {
|
|
if (strcmp(action_buf, "ignore") == 0) {
|
|
state = STATE_BEGINING;
|
|
buf_index = 0;
|
|
*cmd_buf = '\0';
|
|
res = action_append(
|
|
expr_buf,
|
|
action_buf,
|
|
cmd_buf);
|
|
if (res < 0)
|
|
error = abs(res);
|
|
} else
|
|
error = 1;
|
|
} else if (isalpha(c))
|
|
add_to_buff(action_buf, buf_index, c);
|
|
else
|
|
error = 1;
|
|
break;
|
|
|
|
case STATE_COMMAND:
|
|
if (c == '\n' || c == '\0') {
|
|
state = STATE_BEGINING;
|
|
buf_index = 0;
|
|
res = action_append(
|
|
expr_buf,
|
|
action_buf,
|
|
cmd_buf);
|
|
if (res < 0)
|
|
error = abs(res);
|
|
} else
|
|
add_to_buff(cmd_buf, buf_index, c);
|
|
break;
|
|
}
|
|
|
|
if (!error && (!eof || i < byte_cnt))
|
|
continue;
|
|
|
|
switch (state) {
|
|
case STATE_DIRECTIVE:
|
|
if (error == 7)
|
|
fprintf(stderr, "Directive is too long "
|
|
"(line %d)\n", line);
|
|
else if (error == 8)
|
|
fprintf(stderr, "Unknown directive "
|
|
"(line %d)\n", line);
|
|
else
|
|
fprintf(stderr, "Only letters are "
|
|
"allowed in directive "
|
|
"name (line %d)\n",
|
|
line);
|
|
break;
|
|
|
|
case STATE_EXPRESSION_START:
|
|
case STATE_EXPRESSION:
|
|
if (error == 7)
|
|
fprintf(stderr, "Expression is too long "
|
|
"(line %d)\n", line);
|
|
else
|
|
fprintf(stderr, "Expression must be "
|
|
"enclosed with quotes "
|
|
"\" (line %d)\n", line);
|
|
break;
|
|
|
|
case STATE_EXPRESSION_END:
|
|
fprintf(stderr, "One space/tab and "
|
|
"action must follow "
|
|
"each expression"
|
|
" (line %d)\n", line);
|
|
break;
|
|
|
|
case STATE_ACTION:
|
|
if (error == 1)
|
|
fprintf(stderr, "Only letters are "
|
|
"allowed in action "
|
|
"name (line %d)\n",
|
|
line);
|
|
else if (error == 101)
|
|
fprintf(stderr, "Invalid expression "
|
|
"(line %d)\n", line);
|
|
else if (error == 7)
|
|
fprintf(stderr, "Action is too long "
|
|
"(line %d)\n", line);
|
|
else
|
|
fprintf(stderr, "Invalid action "
|
|
"(line %d)\n", line);
|
|
break;
|
|
|
|
case STATE_VALUE:
|
|
if (error == 1)
|
|
fprintf(stderr, "Invalid directive "
|
|
"(line %d)\n", line);
|
|
else if (error == 7)
|
|
fprintf(stderr, "Value is too long "
|
|
"(line %d)\n", line);
|
|
else
|
|
fprintf(stderr, "Invalid value "
|
|
"(line %d)\n", line);
|
|
break;
|
|
|
|
case STATE_COMMAND:
|
|
if (error == 101)
|
|
fprintf(stderr, "Invalid expression "
|
|
"(line %d)\n", line);
|
|
else if (error == 7)
|
|
fprintf(stderr, "Command is too long "
|
|
"(line %d)\n", line);
|
|
else
|
|
fprintf(stderr, "Invalid action "
|
|
"(line %d)\n", line);
|
|
break;
|
|
|
|
case STATE_COMMENT:
|
|
if (!eof)
|
|
fprintf(stderr, "Unknown error "
|
|
"(line %d)\n", line);
|
|
else
|
|
error = 0;
|
|
break;
|
|
|
|
case STATE_BEGINING:
|
|
/* everything is OK */
|
|
error = 0;
|
|
break;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
close(f_rc);
|
|
|
|
return_val(error);
|
|
}
|
|
|
|
|
|
/*
|
|
* free rcconfig data
|
|
*/
|
|
void
|
|
rcconfig_free()
|
|
{
|
|
GList *list_node;
|
|
|
|
begin_func("rcconfig_free");
|
|
|
|
list_node = action_list;
|
|
while (list_node) {
|
|
ACTION *action = list_node->data;
|
|
|
|
g_free(action->command);
|
|
regfree(&action->expression);
|
|
g_free(action);
|
|
list_node = list_node->next;
|
|
}
|
|
g_list_free(action_list);
|
|
|
|
return_void();
|
|
}
|