-
Notifications
You must be signed in to change notification settings - Fork 14
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Create Crypto binaries reporting tool #118
base: release/202311
Are you sure you want to change the base?
Changes from all commits
a4c0b6c
7e45e45
e857973
ffdc2b6
820e223
8706b8e
82b76dc
7ab9d59
bff6eeb
b0fb1ac
c8c79b1
65fe4be
fb37990
f7c44d0
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||||||
---|---|---|---|---|---|---|---|---|
@@ -0,0 +1,289 @@ | ||||||||
from pathlib import Path | ||||||||
import argparse | ||||||||
import subprocess | ||||||||
# | ||||||||
# Setup and parse arguments. | ||||||||
# | ||||||||
parser = argparse.ArgumentParser() | ||||||||
parser.add_argument("-s", "--source", dest="source", type=str, required=True) # Source report file | ||||||||
parser.add_argument("-t", "--target", dest="target", type=str, required=True) # Target report file | ||||||||
args = parser.parse_args() | ||||||||
|
||||||||
# | ||||||||
# Initialize dictionaries to store data from the reports. | ||||||||
# | ||||||||
|
||||||||
# basic report information | ||||||||
source_report_info = {"flavor": "", "build_target": "", "Build Time": "", "tool_chain": ""} | ||||||||
target_report_info = {"flavor": "", "build_target": "", "Build Time": "", "tool_chain": ""} | ||||||||
|
||||||||
# submodules information | ||||||||
source_submodules = {} | ||||||||
target_submodules = {} | ||||||||
|
||||||||
# binaries sizes per architecture | ||||||||
source_binary_sizes_per_arch = {} | ||||||||
target_binary_sizes_per_arch = {} | ||||||||
|
||||||||
# openssl linked lib for binary per arh | ||||||||
source_linked_openssllib_per_arch = {} | ||||||||
target_linked_openssllib_per_arch = {} | ||||||||
|
||||||||
# openssl configuration flags | ||||||||
source_openssl_conf_report = {} | ||||||||
target_openssl_conf_report = {} | ||||||||
|
||||||||
size_change_threshold = 10 # in percentage % | ||||||||
|
||||||||
def log_warning(msg): | ||||||||
subprocess.call(["echo", f"##vso[task.logissue type=warning;]{msg}"]) | ||||||||
#print(f"##[warning]{msg}\n") | ||||||||
|
||||||||
def log_section(msg): | ||||||||
print(f"##[section]{msg}\n") | ||||||||
|
||||||||
def parse_report(report_file, report_dict): | ||||||||
|
||||||||
with report_file.open() as f: | ||||||||
lines = f.readlines() | ||||||||
|
||||||||
record_sizes = False | ||||||||
record_linked_openssllib = False | ||||||||
|
||||||||
for line in lines: | ||||||||
|
||||||||
# get basic report information | ||||||||
if "FLAVOR" in line: | ||||||||
flavor = line.split("-")[1].strip().split()[0] | ||||||||
report_dict["info"]["flavor"] = flavor | ||||||||
build_target = line.split("-")[2].strip() | ||||||||
report_dict["info"]["build_target"] = build_target | ||||||||
|
||||||||
elif "Build Time" in line: | ||||||||
report_dict["info"]["Build Time"] = line.split(":")[1].strip() | ||||||||
|
||||||||
elif "Tool Chain" in line: | ||||||||
report_dict["info"]["tool_chain"] = line.split(":")[1].strip() | ||||||||
|
||||||||
elif "ARCH" in line: | ||||||||
arch = line.split(":")[1].strip() | ||||||||
report_dict["sizes_per_arch"][arch] = {} | ||||||||
report_dict["linked_openssllib_per_arch"][arch] = {} | ||||||||
|
||||||||
elif "Crypto binaries sizes report" in line: | ||||||||
record_sizes = True | ||||||||
|
||||||||
elif "Linked Openssl configuration" in line: | ||||||||
record_sizes = False | ||||||||
record_linked_openssllib = True | ||||||||
|
||||||||
# crypto binaries info | ||||||||
elif ".efi" in line: | ||||||||
binary_name = line.split("-")[0].strip() | ||||||||
# sizes info | ||||||||
if record_sizes: | ||||||||
sizes = line.split("-")[1].strip() | ||||||||
uncompressed_size = sizes.split("|")[0].split(":")[1].strip() | ||||||||
compressed_size = sizes.split("|")[1].split(":")[1].strip() | ||||||||
|
||||||||
report_dict["sizes_per_arch"][arch][binary_name] = {"Uncompressed size": uncompressed_size, "LZMA compressed size": compressed_size} | ||||||||
|
||||||||
# linked openssl lib info | ||||||||
if record_linked_openssllib: | ||||||||
openssllib = line.split("-")[1].split(":")[1].strip() | ||||||||
report_dict["linked_openssllib_per_arch"][arch][binary_name] = openssllib | ||||||||
|
||||||||
elif "File: OpensslLib" in line: | ||||||||
current_openssl_lib_file = line.split(":")[1].strip() | ||||||||
report_dict["openssl_libs_flags"][current_openssl_lib_file] = {"OPENSSL_FLAGS": [], "OPENSSL_FLAGS_CONFIG": []} | ||||||||
|
||||||||
elif "DEFINE OPENSSL_FLAGS" in line: | ||||||||
define_openssl_flags = line.split("=")[0].strip().split()[1].strip() | ||||||||
flags_list = line.split("=")[1].strip().split() | ||||||||
report_dict["openssl_libs_flags"][current_openssl_lib_file][define_openssl_flags] = flags_list | ||||||||
|
||||||||
# submodules info | ||||||||
elif "Name:" in line: | ||||||||
current_submodule = line.split(":")[1].strip() | ||||||||
report_dict["submodules"][current_submodule] = {"Branch": "", "Commit": ""} | ||||||||
elif "Branch:" in line: | ||||||||
report_dict["submodules"][current_submodule]["Branch"] = line.split(":")[1].strip() | ||||||||
elif "Commit:" in line: | ||||||||
report_dict["submodules"][current_submodule]["Commit"] = line.split(":")[1].strip() | ||||||||
|
||||||||
def compare_sizes(source_sizes, target_sizes): | ||||||||
|
||||||||
# compare binaries sizes per architecture | ||||||||
for arch in source_sizes: | ||||||||
log_section(f"Comparing binary sizes for Arch {arch}:") | ||||||||
if arch not in target_sizes: | ||||||||
log_warning(f"Architecture {arch} not found in target report!") | ||||||||
continue | ||||||||
for binary in source_sizes[arch]: | ||||||||
if binary not in target_sizes[arch]: | ||||||||
log_warning(f"A new binary has been added! - {binary} is not found in target report for architecture {arch}") # log warning - New Binary! | ||||||||
continue | ||||||||
source_binary_size = source_sizes[arch][binary] | ||||||||
target_binary_size = target_sizes[arch][binary] | ||||||||
|
||||||||
# get compressed and uncompressed sizes | ||||||||
source_binary_size_uncompressed = float(source_binary_size['Uncompressed size'].split()[0]) | ||||||||
source_binary_size_compressed = float(source_binary_size['LZMA compressed size'].split()[0]) | ||||||||
target_binary_size_uncompressed = float(target_binary_size['Uncompressed size'].split()[0]) | ||||||||
target_binary_size_compressed = float(target_binary_size['LZMA compressed size'].split()[0]) | ||||||||
|
||||||||
# calculate size difference in percentage | ||||||||
size_diff_uncompressed = round((source_binary_size_uncompressed / target_binary_size_uncompressed - 1) * 100, 2) | ||||||||
size_diff_compressed = round((source_binary_size_compressed / target_binary_size_compressed - 1) * 100, 2) | ||||||||
|
||||||||
# check increase & decrease in uncompressed size | ||||||||
if size_diff_uncompressed >= size_change_threshold: # check increase in size | ||||||||
log_warning(f"Uncompressed size for {binary} in architecture {arch} has increased by {size_diff_uncompressed}%!") # log warning - Increase in size! | ||||||||
if size_diff_uncompressed <= (-size_change_threshold): # check decrease in size | ||||||||
log_warning(f"Uncompressed size for {binary} in architecture {arch} has decreased by {size_diff_uncompressed}%!") # log warning - Decrease in size! | ||||||||
|
||||||||
# check increase & decrease in compressed size | ||||||||
if size_diff_compressed >= size_change_threshold: | ||||||||
log_warning( | ||||||||
f"Compressed size for {binary} in architecture {arch} has increased by {size_diff_compressed}%!") # log warning - Increase in size! | ||||||||
if size_diff_compressed <= (-size_change_threshold): | ||||||||
log_warning( | ||||||||
f"Compressed size for {binary} in architecture {arch} has decreased by {size_diff_compressed}%!") # log warning - Decrease in size! | ||||||||
|
||||||||
# check if a binary is missing in source report | ||||||||
for binary in target_sizes[arch]: | ||||||||
if binary not in source_sizes[arch]: | ||||||||
log_warning(f"A binary has been removed! - {binary} is not found in source report for architecture {arch}") # log warning - Missing Binary! | ||||||||
|
||||||||
def comapre_linked_openssllib(source_linked_openssllib, target_linked_openssllib): | ||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Suggested change
|
||||||||
|
||||||||
for arch in source_linked_openssllib: | ||||||||
log_section(f"Comparing linked OpensslLib for Arch: {arch}:") | ||||||||
|
||||||||
for binary in source_linked_openssllib[arch]: | ||||||||
if binary not in target_linked_openssllib[arch]: | ||||||||
# already reported new/missing binaries in the sizes comparison, so no need to report them again here | ||||||||
continue | ||||||||
|
||||||||
source_linked_openssllib_for_binary = source_linked_openssllib[arch][binary] | ||||||||
target_linked_openssllib_for_binary = target_linked_openssllib[arch][binary] | ||||||||
|
||||||||
if source_linked_openssllib_for_binary != target_linked_openssllib_for_binary: | ||||||||
log_warning( | ||||||||
f"Linked OpensslLib has been changed for {binary} in architecture {arch}! - New {source_linked_openssllib_for_binary}, Old {target_linked_openssllib_for_binary}") # log warning - Change in linked OpensslLib*.inf! | ||||||||
|
||||||||
def compare_openssl_flags(source_flags, target_flags): | ||||||||
|
||||||||
for file in source_flags: | ||||||||
log_section(f"Comparing OpensslLib configuration flags for file {file}:") | ||||||||
if file not in target_flags: | ||||||||
log_warning(f"A new OpensslLib*.inf has been added! - {file} is not found in target report") # log warning - New OpensslLib*.inf file! | ||||||||
continue | ||||||||
|
||||||||
# compare "OPENSSL_FLAGS" | ||||||||
source_flags_for_file = source_flags[file]["OPENSSL_FLAGS"] | ||||||||
target_flags_for_file = target_flags[file]["OPENSSL_FLAGS"] | ||||||||
|
||||||||
for flag in source_flags_for_file: | ||||||||
if flag not in target_flags_for_file: | ||||||||
log_warning(f"A new flag has been added in {file}! - In OPENSSL_FLAGS section, flag {flag} is not found in target report") # log warning - New flag in OpensslLib*.inf file! | ||||||||
|
||||||||
for flag in target_flags_for_file: | ||||||||
if flag not in source_flags_for_file: | ||||||||
log_warning(f"A flag has been removed in {file}! - In OPENSSL_FLAGS section, flag {flag} is not found in source report") # log warning - Missing flag in OpensslLib*.inf file! | ||||||||
|
||||||||
# compare "OPENSSL_FLAGS_CONFIG" | ||||||||
source_configflags_for_file = source_flags[file]["OPENSSL_FLAGS_CONFIG"] | ||||||||
target_configflags_for_file = target_flags[file]["OPENSSL_FLAGS_CONFIG"] | ||||||||
|
||||||||
for flag in source_configflags_for_file: | ||||||||
if flag not in target_configflags_for_file: | ||||||||
log_warning(f"A new flag has been added in {file}! - In OPENSSL_FLAGS_CONFIG section, flag {flag} is not found in target report") # log warning - New flag in OpensslLib*.inf file! | ||||||||
|
||||||||
for flag in target_configflags_for_file: | ||||||||
if flag not in source_configflags_for_file: | ||||||||
log_warning(f"A flag has been removed in {file}! - In OPENSSL_FLAGS_CONFIG section, flag {flag} is not found in source report") # log warning - Missing flag in OpensslLib*.inf file! | ||||||||
|
||||||||
for file in target_flags: | ||||||||
if file not in source_flags: | ||||||||
log_warning(f"An OpensslLib*.inf has been removed! - {file} is not found in source report") # log warning - Missing OpensslLib*.inf file! | ||||||||
|
||||||||
def compare_submodules(source_submodules, target_submodules): | ||||||||
|
||||||||
log_section("Comparing submodules") | ||||||||
for submodule in source_submodules: | ||||||||
if submodule not in target_submodules: | ||||||||
log_warning(f"A new submodule has been added! - {submodule} is not found in target report") # log warning - New submodule! | ||||||||
continue | ||||||||
|
||||||||
source_branch = source_submodules[submodule]["Branch"] | ||||||||
target_branch = target_submodules[submodule]["Branch"] | ||||||||
source_commit = source_submodules[submodule]["Commit"] | ||||||||
target_commit = target_submodules[submodule]["Commit"] | ||||||||
|
||||||||
if source_branch != target_branch: | ||||||||
log_warning(f"Submodule {submodule} has changed branch! - New {source_branch}, Old {target_branch}") # log warning - Change in branch! | ||||||||
|
||||||||
if source_commit != target_commit: | ||||||||
log_warning(f"Submodule {submodule} has changed commit! - New {source_commit}, Old {target_commit}") # log warning - Change in commit! | ||||||||
|
||||||||
for submodule in target_submodules: | ||||||||
if submodule not in source_submodules: | ||||||||
log_warning(f"A submodule has been removed! - {submodule} is not found in source report") # log warning - Missing submodule! | ||||||||
|
||||||||
def compare_reports(source_reports, target_reports): | ||||||||
|
||||||||
# print and compare basic report information | ||||||||
print("Source Branch Report Info:\n") | ||||||||
print(f"Flavor: {source_reports['info']['flavor']}\n") | ||||||||
print(f"Build Target: {source_reports['info']['build_target']}\n") | ||||||||
print(f"Tool Chain: {source_reports['info']['tool_chain']}\n") | ||||||||
print(f"Build Time: {source_reports['info']['Build Time']}\n\n") | ||||||||
|
||||||||
print("Target Branch Report Info:\n") | ||||||||
print(f"Flavor: {target_reports['info']['flavor']}\n") | ||||||||
print(f"Build Target: {target_reports['info']['build_target']}\n") | ||||||||
print(f"Tool Chain: {target_reports['info']['tool_chain']}\n") | ||||||||
print(f"Build Time: {target_reports['info']['Build Time']}\n\n") | ||||||||
|
||||||||
if(source_reports['info']['flavor'] != target_reports['info']['flavor']): | ||||||||
log_warning("Reports are for different flavors!") | ||||||||
|
||||||||
if(source_reports['info']['build_target'] != target_reports['info']['build_target']): | ||||||||
log_warning("Reports are for different build targets!") | ||||||||
|
||||||||
# compare submodules | ||||||||
compare_submodules(source_reports['submodules'], target_reports['submodules']) | ||||||||
|
||||||||
# compare binary sizes per architecture | ||||||||
compare_sizes(source_reports['sizes_per_arch'], target_reports['sizes_per_arch']) | ||||||||
|
||||||||
# compare linked openssl lib for binary per architecture | ||||||||
comapre_linked_openssllib(source_reports['linked_openssllib_per_arch'], target_reports['linked_openssllib_per_arch']) | ||||||||
|
||||||||
# compare openssl configuration flags | ||||||||
compare_openssl_flags(source_reports['openssl_libs_flags'], target_reports['openssl_libs_flags']) | ||||||||
|
||||||||
log_section("Comparison completed successfully!") | ||||||||
|
||||||||
|
||||||||
if __name__ == '__main__': | ||||||||
|
||||||||
source_report_file = Path(args.source) | ||||||||
target_report_file = Path(args.target) | ||||||||
|
||||||||
# wrap the parsed information in single dict | ||||||||
source_reports = {"info": source_report_info, "sizes_per_arch": source_binary_sizes_per_arch, "linked_openssllib_per_arch": source_linked_openssllib_per_arch, "openssl_libs_flags": source_openssl_conf_report, "submodules": source_submodules} | ||||||||
target_reports = {"info": target_report_info, "sizes_per_arch": target_binary_sizes_per_arch, "linked_openssllib_per_arch": target_linked_openssllib_per_arch, "openssl_libs_flags": target_openssl_conf_report, "submodules": target_submodules} | ||||||||
|
||||||||
# read the source report file | ||||||||
log_section(f"Parsing source report file: {source_report_file}") | ||||||||
parse_report(source_report_file, source_reports) | ||||||||
|
||||||||
# read the target report file | ||||||||
log_section(f"Parsing target report file: {target_report_file}") | ||||||||
parse_report(target_report_file, target_reports) | ||||||||
|
||||||||
# compare the reports | ||||||||
log_section("Comparing reports") | ||||||||
compare_reports(source_reports, target_reports) |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,80 @@ | ||
#### RANDOM REPORT - NO REAL DATA! #### | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I would like the actual openssl submodule commit to be linked in the report as well. Any changes there needs to be monitored. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Good idea. I'll add submodules info as well. |
||
|
||
CRYPTO BINARIES REPORT - STANDARD FLAVOR - DEBUG | ||
|
||
Build Time: 2024-12-19 14:31:31 | ||
Tool Chain: VS2022 | ||
============================================= | ||
<------Submodules------> | ||
-------- | ||
Name: MU_BASECORE | ||
Branch: release/202302 | ||
Commit: 51a3e0df37ba26bce7bd89895bb303c256560e00 | ||
-------- | ||
Name: Silicon/Arm/MU_TIANO | ||
Branch: release/202302 | ||
Commit: 966f16aaba56846b01c3fff348f0d29714422f42 | ||
-------- | ||
Name: OpensslPkg/Library/OpensslLib/openssl | ||
Branch: master | ||
Commit: acdbf8e1e4749ad65c51b6a1d0d769ae689404ba | ||
-------- | ||
Name: Common | ||
Branch: release/202302 | ||
Commit: 5fb0d0462731513e3a28d5c0be1932a765dfe03d | ||
============================================= | ||
-------------------------------- | ||
ARCH: X64 | ||
|
||
<------Crypto binaries sizes report------> | ||
|
||
BaseCryptLibUnitTestApp.efi - Uncompressed: 873.00 KB | LZMA Compressed: 370.96 KB | ||
CryptoDxe.efi - Uncompressed: 900.50 KB | LZMA Compressed: 375.64 KB | ||
CryptoPei.efi - Uncompressed: 650.00 KB | LZMA Compressed: 300.14 KB | ||
CryptoRuntimeDxe.efi - Uncompressed: 792.50 KB | LZMA Compressed: 307.67 KB | ||
CryptoSmm.efi - Uncompressed: 534.00 KB | LZMA Compressed: 201.72 KB | ||
CryptoStandaloneMm.efi - Uncompressed: 534.50 KB | LZMA Compressed: 201.75 KB | ||
|
||
<------Linked Openssl configuration------> | ||
|
||
BaseCryptLibUnitTestApp.efi - Linked OpensslLib: OpensslPkg/Library/OpensslLib/OpensslLibFull.inf | ||
CryptoDxe.efi - Linked OpensslLib: OpensslPkg/Library/OpensslLib/OpensslLibFull.inf | ||
CryptoPei.efi - Linked OpensslLib: OpensslPkg/Library/OpensslLib/OpensslLibAccel.inf | ||
CryptoRuntimeDxe.efi - Linked OpensslLib: OpensslPkg/Library/OpensslLib/OpensslLib.inf | ||
CryptoSmm.efi - Linked OpensslLib: OpensslPkg/Library/OpensslLib/OpensslLib.inf | ||
CryptoStandaloneMm.efi - Linked OpensslLib: OpensslPkg/Library/OpensslLib/OpensslLib.inf | ||
-------------------------------- | ||
ARCH: IA32 | ||
|
||
<------Crypto binaries sizes report------> | ||
|
||
CryptoDxe.efi - Uncompressed: 700.00 KB | LZMA Compressed: 320.77 KB | ||
CryptoPei.efi - Uncompressed: 500.62 KB | LZMA Compressed: 250.07 KB | ||
CryptoRuntimeDxe.efi - Uncompressed: 619.00 KB | LZMA Compressed: 281.23 KB | ||
CryptoSmm.efi - Uncompressed: 415.50 KB | LZMA Compressed: 186.04 KB | ||
|
||
<------Linked Openssl configuration------> | ||
|
||
CryptoDxe.efi - Linked OpensslLib: OpensslPkg/Library/OpensslLib/OpensslLibFull.inf | ||
CryptoPei.efi - Linked OpensslLib: OpensslPkg/Library/OpensslLib/OpensslLib.inf | ||
CryptoRuntimeDxe.efi - Linked OpensslLib: OpensslPkg/Library/OpensslLib/OpensslLib.inf | ||
CryptoSmm.efi - Linked OpensslLib: OpensslPkg/Library/OpensslLib/OpensslLib.inf | ||
============================================= | ||
<------Openssl configuration report------> | ||
-------- | ||
File: OpensslLib.inf | ||
DEFINE OPENSSL_FLAGS = -DL_ENDIAN -DOPENSSL_SMALL_FOOTPRINT -D_CRT_SECURE_NO_DEPRECATE -D_CRT_NONSTDC_NO_DEPRECATE -DOPENSSL_NO_EC -DOPENSSL_NO_ECDH -DOPENSSL_NO_ECDSA -DOPENSSL_NO_TLS1_3 -DOPENSSL_NO_SM2 -DOPENSSL_NO_ASM | ||
DEFINE OPENSSL_FLAGS_CONFIG = | ||
-------- | ||
File: OpensslLibAccel.inf | ||
DEFINE OPENSSL_FLAGS = -DOPENSSL_SMALL_FOOTPRINT -D_CRT_SECURE_NO_DEPRECATE -D_CRT_NONSTDC_NO_DEPRECATE -DOPENSSL_NO_EC -DOPENSSL_NO_ECDH -DOPENSSL_NO_ECDSA -DOPENSSL_NO_TLS1_3 -DOPENSSL_NO_SM2 | ||
DEFINE OPENSSL_FLAGS_CONFIG = -DOPENSSL_CPUID_OBJ -DSHA1_ASM -DSHA256_ASM -DSHA512_ASM -DAESNI_ASM -DVPAES_ASM -DGHASH_ASM | ||
-------- | ||
File: OpensslLibFull.inf | ||
DEFINE OPENSSL_FLAGS = -DL_ENDIAN -DOPENSSL_SMALL_FOOTPRINT -D_CRT_SECURE_NO_DEPRECATE -D_CRT_NONSTDC_NO_DEPRECATE -DOPENSSL_NO_ASM | ||
DEFINE OPENSSL_FLAGS_CONFIG = -DSHA1_ASM | ||
-------- | ||
File: OpensslLibFullAccel.inf | ||
DEFINE OPENSSL_FLAGS = -DL_ENDIAN -DOPENSSL_SMALL_FOOTPRINT -D_CRT_SECURE_NO_DEPRECATE -D_CRT_NONSTDC_NO_DEPRECATE | ||
DEFINE OPENSSL_FLAGS_CONFIG = -DOPENSSL_CPUID_OBJ -DSHA1_ASM -DSHA256_ASM -DSHA512_ASM -DAESNI_ASM -DVPAES_ASM -DGHASH_ASM | ||
============================================= |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is an okay place to put the report generating code. One thing to consider is how we're going to update the report we're using as a baseline. I think it would make sense as a pipeline or pipeline step to use the completed PRs report as the new baseline. Just something to think about.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
My thought was to produce a report in each release build as part of the pipeline artifacts.
Then in the PR, if the target branch is "Release/*" to pull the latest report generated for that release branch and that will be the baseline. I will appreciate other opinions and suggestions.
Further research is needed on how to public and grab such the report artifact properly
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That would work. The plan is to move to a main branch in the future but pulling the latest generated report for main would be fine.