Blame SOURCES/0009-Analyzer-support-parallel-requests-parsing.patch

de2e0a
From d22ea2df62b6e245eef75d7201b678601bf63e98 Mon Sep 17 00:00:00 2001
de2e0a
From: Justin Stephenson <jstephen@redhat.com>
de2e0a
Date: Fri, 19 Aug 2022 14:44:11 -0400
de2e0a
Subject: [PATCH 9/9] Analyzer: support parallel requests parsing
de2e0a
MIME-Version: 1.0
de2e0a
Content-Type: text/plain; charset=UTF-8
de2e0a
Content-Transfer-Encoding: 8bit
de2e0a
de2e0a
Analyzer code(primarily the list verbose command) needs
de2e0a
changes to handle parsing the necessary lines from
de2e0a
NSS/PAM log files when multiple intermixed/parallel
de2e0a
client requests are sent to SSSD.
de2e0a
de2e0a
Resolves: https://github.com/SSSD/sssd/issues/6307
de2e0a
de2e0a
Reviewed-by: Alexey Tikhonov <atikhono@redhat.com>
de2e0a
Reviewed-by: Pavel Březina <pbrezina@redhat.com>
de2e0a
de2e0a
Reviewed-by: Alexey Tikhonov <atikhono@redhat.com>
de2e0a
---
de2e0a
 src/tools/analyzer/modules/request.py | 119 +++++++++++++++-----------
de2e0a
 1 file changed, 67 insertions(+), 52 deletions(-)
de2e0a
de2e0a
diff --git a/src/tools/analyzer/modules/request.py b/src/tools/analyzer/modules/request.py
de2e0a
index 935e13adc..b9fe3caf8 100644
de2e0a
--- a/src/tools/analyzer/modules/request.py
de2e0a
+++ b/src/tools/analyzer/modules/request.py
de2e0a
@@ -16,7 +16,6 @@ class RequestAnalyzer:
de2e0a
     """
de2e0a
     module_parser = None
de2e0a
     consumed_logs = []
de2e0a
-    done = ""
de2e0a
     list_opts = [
de2e0a
         Option('--verbose', 'Verbose output', bool, '-v'),
de2e0a
         Option('--pam', 'Filter only PAM requests', bool),
de2e0a
@@ -149,58 +148,74 @@ class RequestAnalyzer:
de2e0a
                 print(line)
de2e0a
         return found_results
de2e0a
 
de2e0a
-    def print_formatted(self, line, verbose):
de2e0a
+    def print_formatted_verbose(self, source, patterns):
de2e0a
+        """
de2e0a
+        Parse line and print formatted verbose list_requests output
de2e0a
+
de2e0a
+        Args:
de2e0a
+            source (Reader): source Reader object
de2e0a
+            patterns (list): List of regex patterns to use for
de2e0a
+                matching lines
de2e0a
+        """
de2e0a
+        # Get CID number, and print the basic line first
de2e0a
+        for line in self.matched_line(source, patterns):
de2e0a
+            cid = self.print_formatted(line)
de2e0a
+
de2e0a
+            # Loop through each line with this CID number to extract and
de2e0a
+            # print the verbose data needed
de2e0a
+            verbose_patterns = ["(cache_req_send|cache_req_process_input|"
de2e0a
+                                "cache_req_search_send)"]
de2e0a
+            for cidline in self.matched_line(source, verbose_patterns):
de2e0a
+                plugin = ""
de2e0a
+                name = ""
de2e0a
+                id = ""
de2e0a
+
de2e0a
+                # skip any lines not pertaining to this CID
de2e0a
+                if f"CID#{cid}]" not in cidline:
de2e0a
+                    continue
de2e0a
+                if "refreshed" in cidline:
de2e0a
+                    continue
de2e0a
+                # CR Plugin name
de2e0a
+                if re.search("cache_req_send", cidline):
de2e0a
+                    plugin = cidline.split('\'')[1]
de2e0a
+                # CR Input name
de2e0a
+                elif re.search("cache_req_process_input", cidline):
de2e0a
+                    name = cidline.rsplit('[')[-1]
de2e0a
+                # CR Input id
de2e0a
+                elif re.search("cache_req_search_send", cidline):
de2e0a
+                    id = cidline.rsplit()[-1]
de2e0a
+
de2e0a
+                if plugin:
de2e0a
+                    print("   - " + plugin)
de2e0a
+                if name:
de2e0a
+                    print("       - " + name[:-2])
de2e0a
+                if (id and ("UID" in cidline or "GID" in cidline)):
de2e0a
+                    print("       - " + id)
de2e0a
+
de2e0a
+    def print_formatted(self, line):
de2e0a
         """
de2e0a
         Parse line and print formatted list_requests output
de2e0a
 
de2e0a
         Args:
de2e0a
             line (str): line to parse
de2e0a
-            verbose (bool): If true, enable verbose output
de2e0a
+        Returns:
de2e0a
+            Client ID from printed line, 0 otherwise
de2e0a
         """
de2e0a
-        plugin = ""
de2e0a
-        name = ""
de2e0a
-        id = ""
de2e0a
-
de2e0a
         # exclude backtrace logs
de2e0a
         if line.startswith('   *  '):
de2e0a
-            return
de2e0a
-        fields = line.split("[")
de2e0a
-        cr_field = fields[3][7:]
de2e0a
-        cr = cr_field.split(":")[0][4:]
de2e0a
+            return 0
de2e0a
         if "refreshed" in line:
de2e0a
-            return
de2e0a
-        # CR Plugin name
de2e0a
-        if re.search("cache_req_send", line):
de2e0a
-            plugin = line.split('\'')[1]
de2e0a
-        # CR Input name
de2e0a
-        elif re.search("cache_req_process_input", line):
de2e0a
-            name = line.rsplit('[')[-1]
de2e0a
-        # CR Input id
de2e0a
-        elif re.search("cache_req_search_send", line):
de2e0a
-            id = line.rsplit()[-1]
de2e0a
-        # CID and client process name
de2e0a
-        else:
de2e0a
-            ts = line.split(")")[0]
de2e0a
-            ts = ts[1:]
de2e0a
-            fields = line.split("[")
de2e0a
-            cid = fields[3][4:-9]
de2e0a
-            cmd = fields[4][4:-1]
de2e0a
-            uid = fields[5][4:-1]
de2e0a
-            if not uid.isnumeric():
de2e0a
-                uid = fields[6][4:-1]
de2e0a
-            print(f'{ts}: [uid {uid}] CID #{cid}: {cmd}')
de2e0a
-
de2e0a
-        if verbose:
de2e0a
-            if plugin:
de2e0a
-                print("   - " + plugin)
de2e0a
-            if name:
de2e0a
-                if cr not in self.done:
de2e0a
-                    print("       - " + name[:-2])
de2e0a
-                    self.done = cr
de2e0a
-            if id:
de2e0a
-                if cr not in self.done:
de2e0a
-                    print("       - " + id)
de2e0a
-                    self.done = cr
de2e0a
+            return 0
de2e0a
+        ts = line.split(")")[0]
de2e0a
+        ts = ts[1:]
de2e0a
+        fields = line.split("[")
de2e0a
+        cid = fields[3][4:-9]
de2e0a
+        cmd = fields[4][4:-1]
de2e0a
+        uid = fields[5][4:-1]
de2e0a
+        if not uid.isnumeric():
de2e0a
+            uid = fields[6][4:-1]
de2e0a
+        print(f'{ts}: [uid {uid}] CID #{cid}: {cmd}')
de2e0a
+        return cid
de2e0a
 
de2e0a
     def list_requests(self, args):
de2e0a
         """
de2e0a
@@ -215,20 +230,20 @@ class RequestAnalyzer:
de2e0a
         # Log messages matching the following regex patterns contain
de2e0a
         # the useful info we need to produce list output
de2e0a
         patterns = [r'\[cmd']
de2e0a
-        patterns.append("(cache_req_send|cache_req_process_input|"
de2e0a
-                        "cache_req_search_send)")
de2e0a
         if args.pam:
de2e0a
             component = source.Component.PAM
de2e0a
             resp = "pam"
de2e0a
 
de2e0a
         logger.info(f"******** Listing {resp} client requests ********")
de2e0a
         source.set_component(component, False)
de2e0a
-        self.done = ""
de2e0a
-        for line in self.matched_line(source, patterns):
de2e0a
-            if isinstance(source, Journald):
de2e0a
-                print(line)
de2e0a
-            else:
de2e0a
-                self.print_formatted(line, args.verbose)
de2e0a
+        if args.verbose:
de2e0a
+            self.print_formatted_verbose(source, patterns)
de2e0a
+        else:
de2e0a
+            for line in self.matched_line(source, patterns):
de2e0a
+                if isinstance(source, Journald):
de2e0a
+                    print(line)
de2e0a
+                else:
de2e0a
+                    self.print_formatted(line)
de2e0a
 
de2e0a
     def track_request(self, args):
de2e0a
         """
de2e0a
-- 
de2e0a
2.37.1
de2e0a