e000aa4
From ba0520650ae7f9f63e48ba9fb3a94297aebe2d0c Mon Sep 17 00:00:00 2001
8167548
From: Martin Sehnoutka <msehnout@redhat.com>
e000aa4
Date: Wed, 7 Sep 2016 14:22:21 +0200
Ondřej Lysoněk 7c0626d
Subject: [PATCH 14/59] Add support for square brackets in ls.
8167548
8167548
---
8167548
 ls.c | 222 +++++++++++++++++++++++++++++++++++++++++++++----------------------
8167548
 1 file changed, 150 insertions(+), 72 deletions(-)
8167548
8167548
diff --git a/ls.c b/ls.c
e000aa4
index 616b2d9..b840136 100644
8167548
--- a/ls.c
8167548
+++ b/ls.c
8167548
@@ -246,7 +246,7 @@ vsf_filename_passes_filter(const struct mystr* p_filename_str,
Jiri Skala 09c09a5
   int ret = 0;
Jiri Skala 09c09a5
   char last_token = 0;
Jiri Skala 09c09a5
   int must_match_at_current_pos = 1;
e000aa4
-
Jiri Skala 09c09a5
+  int matched = 0;
e000aa4
 
Jiri Skala 09c09a5
   str_copy(&filter_remain_str, p_filter_str);
e000aa4
 
8167548
@@ -276,7 +276,7 @@ vsf_filename_passes_filter(const struct mystr* p_filename_str,
Jiri Skala 09c09a5
     static struct mystr s_match_needed_str;
Jiri Skala 09c09a5
     /* Locate next special token */
Jiri Skala 09c09a5
     struct str_locate_result locate_result =
Jiri Skala 09c09a5
-      str_locate_chars(&filter_remain_str, "*?{");
Jiri Skala 09c09a5
+      str_locate_chars(&filter_remain_str, "*?{[");
Jiri Skala 09c09a5
     (*iters)++;
Jiri Skala 09c09a5
     /* Isolate text leading up to token (if any) - needs to be matched */
Jiri Skala 09c09a5
     if (locate_result.found)
8167548
@@ -294,94 +294,172 @@ vsf_filename_passes_filter(const struct mystr* p_filename_str,
Jiri Skala 09c09a5
       str_empty(&filter_remain_str);
Jiri Skala 09c09a5
       last_token = 0;
Jiri Skala 09c09a5
     }
Jiri Skala 09c09a5
-    if (!str_isempty(&s_match_needed_str))
Jiri Skala 09c09a5
-    {
Jiri Skala 09c09a5
-      /* Need to match something.. could be a match which has to start at
Jiri Skala 09c09a5
-       * current position, or we could allow it to start anywhere
Jiri Skala 09c09a5
-       */
Jiri Skala 09c09a5
-      unsigned int indexx;
Jiri Skala 09c09a5
-      locate_result = str_locate_str(&name_remain_str, &s_match_needed_str);
Jiri Skala 09c09a5
-      if (!locate_result.found)
Jiri Skala afd5aab
+
Jiri Skala 09c09a5
+    matched = 0;
Jiri Skala 09c09a5
+    do {
Jiri Skala 09c09a5
+      if (!str_isempty(&s_match_needed_str))
Jiri Skala afd5aab
       {
Jiri Skala afd5aab
-        /* Fail */
Jiri Skala afd5aab
-        goto out;
Jiri Skala 09c09a5
+        if (!matched)
Jiri Skala 09c09a5
+        {
Jiri Skala 09c09a5
+          matched = 1;
Jiri Skala 09c09a5
+        }
Jiri Skala 09c09a5
+        /* Need to match something.. could be a match which has to start at
Jiri Skala 09c09a5
+         * current position, or we could allow it to start anywhere
Jiri Skala 09c09a5
+         */
Jiri Skala 09c09a5
+        unsigned int indexx;
Jiri Skala 09c09a5
+        locate_result = str_locate_str(&name_remain_str, &s_match_needed_str);
Jiri Skala 09c09a5
+        if (!locate_result.found)
Jiri Skala 09c09a5
+        {
Jiri Skala 09c09a5
+          /* Fail */
Jiri Skala 09c09a5
+          goto out;
Jiri Skala 09c09a5
+        }
Jiri Skala 09c09a5
+        indexx = locate_result.index;
Jiri Skala 09c09a5
+        if (must_match_at_current_pos && indexx > 0)
Jiri Skala afd5aab
+        {
Jiri Skala 09c09a5
+          goto out;
Jiri Skala afd5aab
+        }
Jiri Skala 09c09a5
+        if (!must_match_at_current_pos && last_token == 0)
Jiri Skala 09c09a5
+        {
Jiri Skala 09c09a5
+          struct mystr last_str = INIT_MYSTR;
Jiri Skala 09c09a5
+          str_mid_to_end(&name_remain_str, &last_str,
Jiri Skala 09c09a5
+            str_getlen(&name_remain_str) - str_getlen(&s_match_needed_str));
Jiri Skala 09c09a5
+          locate_result = str_locate_str(&last_str, &s_match_needed_str);
Jiri Skala 09c09a5
+          str_free(&last_str);
Jiri Skala 09c09a5
+
Jiri Skala 09c09a5
+          if (locate_result.found)
Jiri Skala 09c09a5
+          {
Jiri Skala 09c09a5
+            ret = 1;
Jiri Skala 09c09a5
+          }
Jiri Skala 09c09a5
+          goto out;
Jiri Skala 09c09a5
+        }
Jiri Skala 09c09a5
+        /* Chop matched string out of remainder */
Jiri Skala 09c09a5
+        str_mid_to_end(&name_remain_str, &temp_str,
Jiri Skala 09c09a5
+                       indexx + str_getlen(&s_match_needed_str));
Jiri Skala 09c09a5
+        str_copy(&name_remain_str, &temp_str);
Jiri Skala 09c09a5
       }
Jiri Skala afd5aab
-      indexx = locate_result.index;
Jiri Skala afd5aab
-      if (must_match_at_current_pos && indexx > 0)
Jiri Skala 09c09a5
+      if (last_token == '?')
Jiri Skala 09c09a5
       {
Jiri Skala 09c09a5
-        goto out;
Jiri Skala 09c09a5
+        if (str_isempty(&name_remain_str))
Jiri Skala 09c09a5
+        {
Jiri Skala 09c09a5
+          goto out;
Jiri Skala 09c09a5
+        }
Jiri Skala 09c09a5
+        str_right(&name_remain_str, &temp_str, str_getlen(&name_remain_str) - 1);
Jiri Skala 09c09a5
+        str_copy(&name_remain_str, &temp_str);
Jiri Skala 09c09a5
+        must_match_at_current_pos = 1;
Jiri Skala 09c09a5
       }
Jiri Skala afd5aab
-      if (!must_match_at_current_pos && last_token == 0)
Jiri Skala 09c09a5
+      else if (last_token == '{')
Jiri Skala 09c09a5
       {
Jiri Skala afd5aab
-        struct mystr last_str = INIT_MYSTR;
Jiri Skala afd5aab
-        str_mid_to_end(&name_remain_str, &last_str,
Jiri Skala afd5aab
-          str_getlen(&name_remain_str) - str_getlen(&s_match_needed_str));
Jiri Skala afd5aab
-        locate_result = str_locate_str(&last_str, &s_match_needed_str);
Jiri Skala afd5aab
-        str_free(&last_str);
Jiri Skala 09c09a5
+        struct str_locate_result end_brace =
Jiri Skala 09c09a5
+          str_locate_char(&filter_remain_str, '}');
Jiri Skala 09c09a5
+        must_match_at_current_pos = 1;
Jiri Skala 09c09a5
+        if (end_brace.found)
Jiri Skala 09c09a5
+        {
Jiri Skala afd5aab
+          int entire = (*iters == 1 && last_token == '{');
Jiri Skala afd5aab
 
Jiri Skala afd5aab
-        if (locate_result.found)
Jiri Skala 09c09a5
+          str_split_char(&filter_remain_str, &temp_str, '}');
Jiri Skala 09c09a5
+          str_copy(&brace_list_str, &filter_remain_str);
Jiri Skala 09c09a5
+          str_copy(&filter_remain_str, &temp_str);
Jiri Skala 09c09a5
+          str_split_char(&brace_list_str, &temp_str, ',');
Jiri Skala 09c09a5
+          while (!str_isempty(&brace_list_str))
Jiri Skala afd5aab
+          {
Jiri Skala 09c09a5
+            str_empty(&new_filter_str);
Jiri Skala afd5aab
+            if (!matched && !entire)
Jiri Skala 09c09a5
+            {
Jiri Skala 09c09a5
+              str_append_char(&new_filter_str, '*');
Jiri Skala 09c09a5
+            }
Jiri Skala 09c09a5
+            str_append_str(&new_filter_str, &brace_list_str);
Jiri Skala 09c09a5
+            str_append_str(&new_filter_str, &filter_remain_str);
Jiri Skala 09c09a5
+            if (vsf_filename_passes_filter(&name_remain_str, &new_filter_str,
Jiri Skala 09c09a5
+                                           iters))
Jiri Skala 09c09a5
+            {
Jiri Skala 09c09a5
+              ret = 1;
Jiri Skala 09c09a5
+              goto out;
Jiri Skala 09c09a5
+            }
Jiri Skala 09c09a5
+            str_copy(&brace_list_str, &temp_str);
Jiri Skala 09c09a5
+            str_split_char(&brace_list_str, &temp_str, ',');
Jiri Skala afd5aab
+          }
Jiri Skala 09c09a5
+          goto out;
Jiri Skala 09c09a5
+        }
Jiri Skala 09c09a5
+        else if (str_isempty(&name_remain_str) ||
Jiri Skala 09c09a5
+                 str_get_char_at(&name_remain_str, 0) != '{')
Jiri Skala afd5aab
         {
Jiri Skala afd5aab
-          ret = 1;
Jiri Skala 09c09a5
+          goto out;
Jiri Skala 09c09a5
+        }
Jiri Skala 09c09a5
+        else
Jiri Skala 09c09a5
+        {
Jiri Skala 09c09a5
+          str_right(&name_remain_str, &temp_str,
Jiri Skala 09c09a5
+                    str_getlen(&name_remain_str) - 1);
Jiri Skala 09c09a5
+          str_copy(&name_remain_str, &temp_str);
Jiri Skala 09c09a5
         }
Jiri Skala 09c09a5
-        goto out;
Jiri Skala afd5aab
-      }
Jiri Skala afd5aab
-      /* Chop matched string out of remainder */
Jiri Skala afd5aab
-      str_mid_to_end(&name_remain_str, &temp_str,
Jiri Skala afd5aab
-                     indexx + str_getlen(&s_match_needed_str));
Jiri Skala afd5aab
-      str_copy(&name_remain_str, &temp_str);
Jiri Skala afd5aab
-    }
Jiri Skala afd5aab
-    if (last_token == '?')
Jiri Skala afd5aab
-    {
Jiri Skala afd5aab
-      if (str_isempty(&name_remain_str))
Jiri Skala afd5aab
-      {
Jiri Skala afd5aab
-        goto out;
Jiri Skala 09c09a5
       }
Jiri Skala afd5aab
-      str_right(&name_remain_str, &temp_str, str_getlen(&name_remain_str) - 1);
Jiri Skala afd5aab
-      str_copy(&name_remain_str, &temp_str);
Jiri Skala afd5aab
-      must_match_at_current_pos = 1;
Jiri Skala afd5aab
-    }
Jiri Skala afd5aab
-    else if (last_token == '{')
Jiri Skala afd5aab
-    {
Jiri Skala afd5aab
-      struct str_locate_result end_brace =
Jiri Skala afd5aab
-        str_locate_char(&filter_remain_str, '}');
Jiri Skala afd5aab
-      must_match_at_current_pos = 1;
Jiri Skala afd5aab
-      if (end_brace.found)
Jiri Skala 09c09a5
+      else if (last_token == '[')
Jiri Skala 09c09a5
       {
Jiri Skala afd5aab
-        str_split_char(&filter_remain_str, &temp_str, '}');
Jiri Skala afd5aab
-        str_copy(&brace_list_str, &filter_remain_str);
Jiri Skala afd5aab
-        str_copy(&filter_remain_str, &temp_str);
Jiri Skala afd5aab
-        str_split_char(&brace_list_str, &temp_str, ',');
Jiri Skala afd5aab
-        while (!str_isempty(&brace_list_str))
Jiri Skala 09c09a5
+        struct str_locate_result end_sqb =
Jiri Skala 09c09a5
+          str_locate_char(&filter_remain_str, ']');
Jiri Skala 09c09a5
+        must_match_at_current_pos = 1;
Jiri Skala 09c09a5
+        if (end_sqb.found)
8167548
         {
8167548
-          str_copy(&new_filter_str, &brace_list_str);
8167548
-          str_append_str(&new_filter_str, &filter_remain_str);
8167548
-          if (vsf_filename_passes_filter(&name_remain_str, &new_filter_str,
8167548
-                                         iters))
Jiri Skala 09c09a5
+          unsigned int cur_pos;
Jiri Skala 09c09a5
+          char stch, ench;
Jiri Skala 09c09a5
+          const char *p_brace;
Jiri Skala 09c09a5
+
Jiri Skala 09c09a5
+          str_split_char(&filter_remain_str, &temp_str, ']');
Jiri Skala 09c09a5
+          str_copy(&brace_list_str, &filter_remain_str);
Jiri Skala 09c09a5
+          str_copy(&filter_remain_str, &temp_str);
Jiri Skala 09c09a5
+          p_brace = str_getbuf(&brace_list_str);
Jiri Skala 09c09a5
+          for (cur_pos = 0; cur_pos < str_getlen(&brace_list_str);)
Jiri Skala afd5aab
           {
Jiri Skala afd5aab
-            ret = 1;
Jiri Skala afd5aab
-            goto out;
Jiri Skala 09c09a5
+            stch = p_brace[cur_pos];
Jiri Skala 09c09a5
+            // char vers. range
Jiri Skala 09c09a5
+            if (cur_pos + 2 < str_getlen(&brace_list_str) &&
Jiri Skala 09c09a5
+                p_brace[cur_pos+1] == '-')
Jiri Skala 09c09a5
+            {
Jiri Skala 09c09a5
+              ench = p_brace[cur_pos+2];
Jiri Skala 09c09a5
+              cur_pos += 3;
Jiri Skala 09c09a5
+            }
Jiri Skala 09c09a5
+            else
Jiri Skala 09c09a5
+            {
Jiri Skala 09c09a5
+              ench = stch;
Jiri Skala 09c09a5
+              cur_pos++;
Jiri Skala 09c09a5
+            }
Jiri Skala 09c09a5
+            // expand char[s]
Jiri Skala 09c09a5
+            for (;stch <= ench && !str_isempty(&brace_list_str); stch++)
Jiri Skala 09c09a5
+            {
Jiri Skala 09c09a5
+              str_empty(&new_filter_str);
Jiri Skala 09c09a5
+              if (!matched)
Jiri Skala 09c09a5
+              {
Jiri Skala 09c09a5
+                str_append_char(&new_filter_str, '*');
Jiri Skala 09c09a5
+              }
Jiri Skala 09c09a5
+              str_append_char(&new_filter_str, stch);
Jiri Skala 09c09a5
+              str_append_str(&new_filter_str, &filter_remain_str);
Jiri Skala 09c09a5
+              if (vsf_filename_passes_filter(&name_remain_str, &new_filter_str,
Jiri Skala 09c09a5
+                                             iters))
Jiri Skala 09c09a5
+              {
Jiri Skala 09c09a5
+                ret = 1;
Jiri Skala 09c09a5
+                goto out;
Jiri Skala 09c09a5
+              }
Jiri Skala 09c09a5
+            }
Jiri Skala afd5aab
           }
Jiri Skala afd5aab
-          str_copy(&brace_list_str, &temp_str);
Jiri Skala afd5aab
-          str_split_char(&brace_list_str, &temp_str, ',');
Jiri Skala 09c09a5
+          goto out;
Jiri Skala 09c09a5
+        }
Jiri Skala 09c09a5
+        else if (str_isempty(&name_remain_str) ||
Jiri Skala 09c09a5
+                 str_get_char_at(&name_remain_str, 0) != '[')
Jiri Skala 09c09a5
+        {
Jiri Skala 09c09a5
+          goto out;
Jiri Skala 09c09a5
+        }
Jiri Skala 09c09a5
+        else
Jiri Skala 09c09a5
+        {
Jiri Skala 09c09a5
+          str_right(&name_remain_str, &temp_str,
Jiri Skala 09c09a5
+                    str_getlen(&name_remain_str) - 1);
Jiri Skala 09c09a5
+          str_copy(&name_remain_str, &temp_str);
Jiri Skala afd5aab
         }
Jiri Skala afd5aab
-        goto out;
Jiri Skala afd5aab
-      }
Jiri Skala afd5aab
-      else if (str_isempty(&name_remain_str) ||
Jiri Skala afd5aab
-               str_get_char_at(&name_remain_str, 0) != '{')
Jiri Skala afd5aab
-      {
Jiri Skala afd5aab
-        goto out;
Jiri Skala 09c09a5
       }
Jiri Skala 09c09a5
       else
Jiri Skala 09c09a5
       {
Jiri Skala 09c09a5
-        str_right(&name_remain_str, &temp_str,
Jiri Skala 09c09a5
-                  str_getlen(&name_remain_str) - 1);
Jiri Skala 09c09a5
-        str_copy(&name_remain_str, &temp_str);
Jiri Skala 09c09a5
+        must_match_at_current_pos = 0;
Jiri Skala 09c09a5
       }
Jiri Skala 09c09a5
-    }
Jiri Skala 09c09a5
-    else
Jiri Skala 09c09a5
-    {
Jiri Skala 09c09a5
-      must_match_at_current_pos = 0;
Jiri Skala 09c09a5
-    }
Jiri Skala 09c09a5
+    } while (locate_result.found &&
Jiri Skala 09c09a5
+             str_getlen(&name_remain_str) > 0 && last_token != '*');
Jiri Skala 09c09a5
   }
Jiri Skala 09c09a5
   /* Any incoming string left means no match unless we ended on the correct
Jiri Skala 09c09a5
    * type of wildcard.
8167548
-- 
Ondřej Lysoněk 7c0626d
2.14.4
8167548