From f3cf262d20a3f15669ceb01226560954e0f28f52 Mon Sep 17 00:00:00 2001 From: Sebastian Gniazdowski Date: Wed, 31 Aug 2022 19:54:03 +0200 Subject: [PATCH 1/2] A new substitution special string %(sh {shell-command}) that runs an external command, captures its output and substitutes it in place of the %-string. Can be used as follows: bind generic T :echo '%(sh printf "Hi :)")' --- src/argv.c | 23 +++++++++++++++++++++++ 1 file changed, 23 insertions(+) diff --git a/src/argv.c b/src/argv.c index 3a00e6f43..750876dc1 100644 --- a/src/argv.c +++ b/src/argv.c @@ -315,6 +315,29 @@ format_expand_arg(struct format_context *format, const char *name, const char *e return false; return string_format_from(format->buf, &format->bufpos, "%s", value); } + if (!prefixcmp(name, "%(sh")) { + char cbuf[SIZEOF_STR]; + const char *c=""; + char value[SIZEOF_STR]={0}; + const char *cstart = name + STRING_SIZE("%(sh"); + const int clen = end - cstart - 1; + int size; + FILE *cp; + if (end && clen > 0 && string_format(cbuf, "%.*s", clen, cstart)) { + c=cbuf; + while (isspace(*c)) + c++; + } + if (!*c||strlen(c)==8) + return false; + cp=popen(c,"r"); + size=fread(value,1,SIZEOF_STR-1,cp); + if (value[0]==0 || size==0) + return false; + if (value[size-1]=='\n') + value[size-1]='\0'; + return string_format_from(format->buf, &format->bufpos, "%s", value); + } for (i = 0; i < format->vars_size; i++) { if (string_enum_compare(name, vars[i].name, vars[i].namelen)) From 7058dce75622f8f15642e5a8d0361c544d7ea9b2 Mon Sep 17 00:00:00 2001 From: Sebastian Gniazdowski Date: Thu, 1 Sep 2022 03:27:13 +0200 Subject: [PATCH 2/2] Handle embedded %()-substitutions inside %(prompt) and %(sh) Allowed are e.g.: %(prompt %(commit)), etc. and also recursive like %(sh printf %(sh printf)). --- src/argv.c | 73 ++++++++++++++++++++++++++++++++++++++++++++++++++---- 1 file changed, 68 insertions(+), 5 deletions(-) diff --git a/src/argv.c b/src/argv.c index 750876dc1..4fd9b4af3 100644 --- a/src/argv.c +++ b/src/argv.c @@ -300,7 +300,7 @@ format_expand_arg(struct format_context *format, const char *name, const char *e const char *value; const char *msgstart = name + STRING_SIZE("%(prompt"); const int msglen = end - msgstart - 1; - + char *tmp; if (end && msglen > 0 && string_format(msgbuf, "%.*s", msglen, msgstart)) { const char *msg = msgbuf; @@ -310,7 +310,9 @@ format_expand_arg(struct format_context *format, const char *name, const char *e prompt = msg; } - value = read_prompt(prompt); + tmp=argv_format_arg(&argv_env, prompt); + value = read_prompt(tmp); + free(tmp); if (value == NULL) return false; return string_format_from(format->buf, &format->bufpos, "%s", value); @@ -322,6 +324,7 @@ format_expand_arg(struct format_context *format, const char *name, const char *e const char *cstart = name + STRING_SIZE("%(sh"); const int clen = end - cstart - 1; int size; + char *tmp; FILE *cp; if (end && clen > 0 && string_format(cbuf, "%.*s", clen, cstart)) { c=cbuf; @@ -330,7 +333,9 @@ format_expand_arg(struct format_context *format, const char *name, const char *e } if (!*c||strlen(c)==8) return false; - cp=popen(c,"r"); + tmp=argv_format_arg(&argv_env, c); + cp=popen(tmp,"r"); + free(tmp); size=fread(value,1,SIZEOF_STR-1,cp); if (value[0]==0 || size==0) return false; @@ -352,6 +357,64 @@ format_expand_arg(struct format_context *format, const char *name, const char *e return false; } +static const char* +get_closing_brace(const char *input) +{ + const char *s=input,*cur, *tmp; + int idx,level=0,pair_idx; + int level_to_pos[3000]={[0 ... 2998]=-1, -2}; + int pos_to_level[256]={[0 ... 254]=-1, -2}; + int pair_map[3000]={[0 ... 2998]=-1, -2}; + int final_pairs[3000]={[0 ... 2998]=-1, -2}; + int first=-1; + while ( (cur = strstr(s,"(")), (tmp = strstr(s,")")), (cur&&cur 0 ) { + pair_idx=level_to_pos[level]; + pos_to_level[idx]=level --; + if (input[pair_idx]=='(' && input[idx]==')' || + input[pair_idx]==')' && input[idx]=='(' ) { + final_pairs[idx]=pair_idx; + final_pairs[pair_idx]=idx; + } + } else { + pos_to_level[idx]=-1; + } + } + } + + for (idx=0;pos_to_level[idx]!=-2; idx++) { + bool ok=false; + if (pos_to_level[idx]==-1) + continue; + for (int j=0; final_pairs[j]!=-2;j++) { + if (final_pairs[j]==-1) + continue; + if (final_pairs[j]==idx) { + if (idx==0) + first=j; + ok=true; + break; + } + } + if (!ok) + return NULL; + } + + if (pos_to_level[0] < 0) + return NULL; + if (first<0) + return NULL; + + return input+first; +} + + static bool format_append_arg(struct format_context *format, const char ***dst_argv, const char *arg) { @@ -362,7 +425,7 @@ format_append_arg(struct format_context *format, const char ***dst_argv, const c const char *var = strstr(arg, "%("); const char *esc = strstr(arg, "%%"); bool is_escaped = esc && (esc < var || !var); - const char *closing = var && !is_escaped ? strchr(var, ')') : NULL; + const char *closing = var && !is_escaped ? get_closing_brace(var+1) : NULL; const char *next = is_escaped ? esc + 2 : closing ? closing + 1 : NULL; int len = var && !is_escaped ? var - arg : esc ? esc - arg + 1 : strlen(arg); @@ -372,7 +435,7 @@ format_append_arg(struct format_context *format, const char ***dst_argv, const c if (len && !string_format_from(format->buf, &format->bufpos, "%.*s", len, arg)) return false; - if (var && !is_escaped && !format_expand_arg(format, var, next)) + if (var && !is_escaped && !format_expand_arg(format, var, next)) return false; arg = next;