From bd04f401610b3118eb9cfecaf903dbc6e4c0dd47 Mon Sep 17 00:00:00 2001 From: Tejas Manhas Date: Tue, 12 Mar 2024 19:38:44 +0530 Subject: [PATCH] Added the Test_build.py that builds the kernel and calls Build_bisector if build fails Signed-off-by: Tejas Manhas --- OpTestConfiguration.py | 3 +- testcases/Build_bisector.py | 11 ++- testcases/Email_git.py | 147 ++++++++++++++++++++++++++++++++++++ testcases/Test_build.py | 133 ++++++++++++++++++++++++++++++++ 4 files changed, 292 insertions(+), 2 deletions(-) create mode 100644 testcases/Email_git.py create mode 100644 testcases/Test_build.py diff --git a/OpTestConfiguration.py b/OpTestConfiguration.py index 3c5ac4a1..afeae4a7 100644 --- a/OpTestConfiguration.py +++ b/OpTestConfiguration.py @@ -460,7 +460,8 @@ def get_parser(): help="Don't exit if we find unknown command line arguments") misc_group.add_argument("--secvar-payload-url", help="Specify a URL for the secvar test data payload") - + misc_group.add_argument("--bisect-flag", + help="Specify if bisection is to be done or not") return parser diff --git a/testcases/Build_bisector.py b/testcases/Build_bisector.py index 220a760d..694f0e46 100644 --- a/testcases/Build_bisector.py +++ b/testcases/Build_bisector.py @@ -91,12 +91,21 @@ def runTest(self): makefile_path = os.path.join(self.conf.basedir, "make.sh") self.cv_HOST.copy_test_file_to_host(makefile_path, dstdir=self.linux_path) self.connection.run_command("git bisect start") - self.connection.run_command("git bisect good {} ".format(self.good_commit)) + folder_type=re.split(r'[\/\\.]',str(self.repo))[-2] + print("FOLDER",folder_type) + if folder_type == 'linux-next': + self.connection.run_command("git fetch --tags") + good_tag=self.connection.run_command("git tag -l 'v[0-9]*' | sort -V | tail -n 1") + self.connection.run_command("git bisect good {} ".format(good_tag)) + else: + self.connection.run_command("git bisect good {} ".format(self.good_commit)) self.connection.run_command(" git bisect bad ") self.connection.run_command("chmod +x ./make.sh ") commit = self.connection.run_command(" git bisect run ./make.sh") badCommit = [word for word in commit if word.endswith("is the first bad commit")] badCommit= badCommit[0].split()[0] email = self.get_email(badCommit) + log.info("LOGGG") + self.connection.run_command("git bisect log") self.connection.run_command("git bisect reset") return email , badCommit \ No newline at end of file diff --git a/testcases/Email_git.py b/testcases/Email_git.py new file mode 100644 index 00000000..c1dcc79a --- /dev/null +++ b/testcases/Email_git.py @@ -0,0 +1,147 @@ +#Email_git.py: to make boot of the repo and test it through exit code + +#!/usr/bin/env python3 +# OpenPOWER Automated Test Project +# +# Contributors Listed Below - COPYRIGHT 2024 +# [+] International Business Machines Corp. +# +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or +# implied. See the License for the specific language governing +# permissions and limitations under the License. +import json +# import requests +import os +import uuid +from datetime import datetime +import re +import unittest +import os +from urllib.parse import urlparse +from enum import Enum +import subprocess +import OpTestConfiguration +import OpTestLogger +import configparser +import sys +from testcases import BisectKernel +from testcases import Test_build +from common.OpTestSystem import OpSystemState +from common.OpTestSOL import OpSOLMonitorThread +from common.OpTestInstallUtil import InstallUtil +from common.Exceptions import CommandFailed +# from testcases import Boot +log = OpTestLogger.optest_logger_glob.get_logger(__name__) +class Email_git(unittest.TestCase): + """ + Test case for bisecting the Linux kernel using Git Bisect. + This test downloads the Linux kernel from a specified repository, + configures and compiles it, and then uses Git Bisect to find the + commit that introduced a specific issue. + """ + def setUp(self): + """ + Set up the test environment. + Initializes test parameters and checks required configurations. + """ + self.conf = OpTestConfiguration.conf + self.cv_HOST = self.conf.host() + self.cv_SYSTEM = self.conf.system() + self.connection = self.cv_SYSTEM.cv_HOST.get_ssh_connection() + self.console_thread = OpSOLMonitorThread(1, "console") + self.host_cmd_timeout = self.conf.args.host_cmd_timeout + self.repo = self.conf.args.git_repo + self.repo_reference = self.conf.args.git_repo_reference + self.branch = self.conf.args.git_branch + self.home = self.conf.args.git_home + # self.config_path = self.conf.args.git_repoconfigpatih + self.config = self.conf.args.git_repoconfig + self.good_commit = self.conf.args.good_commit + self.bad_commit = self.conf.args.bad_commit + self.bisect_script = self.conf.args.bisect_script + self.bisect_category = self.conf.args.bisect_category + self.append_kernel_cmdline = self.conf.args.append_kernel_cmdline + self.linux_path = os.path.join(self.home, "linux") + if not self.repo: + self.fail("Provide git repo of kernel to install") + if not (self.conf.args.host_ip and self.conf.args.host_user and self.conf.args.host_password): + self.fail( + "Provide host ip user details refer, --host-{ip,user,password}") + def get_commit_message(self,commit_sha): + try: + self.connection.run_command(" if [ '$(pwd)' != {} ]; then cd {} || exit 1 ; fi ".format(self.linux_path,self.linux_path)) + + commit_message = self.connection.run_command("git log -n 1 --pretty=format:%s {} | sed -r 's/\x1B\[[0-9:]*[JKsu]//g'".format(commit_sha)) + print(commit_message) + except subprocess.CalledProcessError: + commit_message = None + print(commit_message[0].strip()) + return commit_message[0].strip() + def runTest(self): + # def generate_email_template(machine_type, gcc_version, commit, error_message): + machine_type = self.connection.run_command("uname -m") + gcc_version = self.connection.run_command("gcc --version")[0] + kernel_version = self.connection.run_command("uname -r") + try: + with open("output.json", "r") as file: + data = json.load(file) + error_message = data.get("error", "") + commit = str(data.get("commit", ""))[:7] + except FileNotFoundError: + print("Error: output.json not found.") + error_message = "" + commit = "" + + fix_description = self.get_commit_message(commit) + # self.repo = "https://git.kernel.org/pub/scm/linux/kernel/git/netdev/net.git" + + if "https://git.kernel.org/pub/scm/linux/kernel/git/netdev/net.git" in self.repo: + linux = "netdev/net" + elif "https://git.kernel.org/pub/scm/linux/kernel/git/netdev/net-next.git" in self.repo: + linux = "netdev/net-next" + elif "https://git.kernel.org/pub/scm/linux/kernel/git/mkp/scsi.git" in self.repo or "git://git.kernel.org/pub/scm/linux/kernel/git/mkp/scsi.git" in self.repo: + linux = "scsi/scsi-queue" + elif "https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git" in self.repo: + linux = "mainline/master" + elif "https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git" in self.repo: + linux = "linux-next/master" + else: + linux = "linux" + + subject = "[{}][bisected {}][PPC {}] build fail with error: {}".format(linux,commit, machine_type, error_message) + + body = """ + Greetings, + + Today's next kernel fails to build with gcc {} on {} machine. + + Kernel build fail at error: {} + + Kernel Version: {} + Machine Type: {} + gcc: {} + Commit: {} + + kernel builds fine when the bad commit ({}) is reverted + {} - {} + + -- + Regards + PPC KBOT + """.format(gcc_version, machine_type, error_message,kernel_version, machine_type, gcc_version, commit,commit, commit, fix_description) + print(subject) + print(body) + + with open("email.json","w") as email: + json.dump({"subject":subject,"body":body},email) + # return subject, body + diff --git a/testcases/Test_build.py b/testcases/Test_build.py new file mode 100644 index 00000000..20f3edc8 --- /dev/null +++ b/testcases/Test_build.py @@ -0,0 +1,133 @@ +#!/usr/bin/env python3 +# OpenPOWER Automated Test Project +# +# Contributors Listed Below - COPYRIGHT 2024 +# [+] International Business Machines Corp. +# +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or +# implied. See the License for the specific language governing +# permissions and limitations under the License. +import json +import os +import re +import unittest +from urllib.parse import urlparse +import subprocess +import OpTestConfiguration +import OpTestLogger +import configparser +import sys +from testcases import Build_bisector +from testcases import Email_git +from common.OpTestSystem import OpSystemState +from common.OpTestSOL import OpSOLMonitorThread +from common.Exceptions import CommandFailed +log = OpTestLogger.optest_logger_glob.get_logger(__name__) +class Test_build(unittest.TestCase): + """ + Test case for building kernel and calling Build_bisector.py in case of build failure + """ + def setUp(self): + """ + Set up the test environment. + Initializes test parameters and checks required configurations. + """ + self.conf = OpTestConfiguration.conf + self.cv_HOST = self.conf.host() + self.cv_SYSTEM = self.conf.system() + self.connection = self.cv_SYSTEM.cv_HOST.get_ssh_connection() + self.console_thread = OpSOLMonitorThread(1, "console") + self.host_cmd_timeout = self.conf.args.host_cmd_timeout + self.repo = self.conf.args.git_repo + self.repo_reference = self.conf.args.git_repo_reference + self.branch = self.conf.args.git_branch + self.home = self.conf.args.git_home + self.config = self.conf.args.git_repoconfig + self.good_commit = self.conf.args.good_commit + self.bad_commit = self.conf.args.bad_commit + self.bisect_script = self.conf.args.bisect_script + self.bisect_category = self.conf.args.bisect_category + self.append_kernel_cmdline = self.conf.args.append_kernel_cmdline + self.linux_path = os.path.join(self.home, "linux") + self.bisect_flag = self.conf.args.bisect_flag + if not self.repo: + self.fail("Provide git repo of kernel to install") + if not (self.conf.args.host_ip and self.conf.args.host_user and self.conf.args.host_password): + self.fail( + "Provide host ip user details refer, --host-{ip,user,password}") + + def build_kernel(self): + """ + Build and install the Linux kernel. + """ + try: + build_command = "make -j && make modules_install && make install" + err=self.connection.run_command(build_command, timeout=self.host_cmd_timeout) + log.info("Kernel build successful") + return 0,err + except CommandFailed as e: + log.error("Kernel build failed: {}".format(e)) + return 4,e + + def Store_loc ( self, er) : + """ + To get location of file in which error is introduced + """ + pattern = r"([\w\d_]+\/(?:(?:[\w\d_]+\/)*[\w\d_]+\b))" + matches = [match.group(1) for match in re.finditer(pattern,er)] + return matches + + def runTest(self): + self.connection.run_command("if [ -d {} ]; then rm -rf {}; fi".format(self.home,self.home)) + self.connection.run_command("if [ ! -d {} ]; then mkdir -p {}; fi".format(self.home,self.home)) + self.connection.run_command("cd {}".format(self.home)) + if not self.branch: + self.branch='master' + log.info("CD DONE") + self.connection.run_command("git clone --depth 1 {} -b {}".format(self.repo, self.branch)) + self.connection.run_command("git clone -b {} {} linux".format( self.branch, self.repo),timeout=3000) + self.connection.run_command("cd linux") + commit = self.connection.run_command(" git log -1 --format=%H | sed -r 's/\x1B\[[0-9:]*[JKsu]//g'") + self.connection.run_command("cd ..") + self.connection.run_command("wget http://ltc-jenkins.aus.stglabs.ibm.com:81/abdul/ioci/kernel_config -o linux/.config") + self.connection.run_command("cd linux && make olddefconfig") + error = self.build_kernel() + log.info("COMMAND RUN") + exit_code = error[0] + errVal = str(error[1]) + log.info("printing the exit code '{}'".format(exit_code)) + entry=[] + if exit_code != 0: + entry = self.Store_loc(errVal)[-1] + badCommit = commit[-1] + if self.bisect_flag == '1': + log.info("BUILD_BISECTOR CALLED") + bisect = Build_bisector.Buil_bisector() + bisect.setUp() + res = bisect.runTest() + log.info("BUILD_BISECTOR END") + emaili=res[0] + commiti=res[1] + log.info("COMMIT REVERT HAS TO BE CHECKED MANUALLY") + else : + emaili="" + commiti=commit[-1] + else : + emaili="" + commiti=commit[-1] + with open('output.json','w') as f: + json.dump({"exit_code":exit_code,"email":emaili,"commit": commiti,"error":entry,"flag":self.bisect_flag},f) + if exit_code != 0: + email = Email_git.Email_git() + email.setUp() + email.runTest() + return exit_code