Skip to content
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

Deduplication Bug in Semgrep JSON Report Causing Mitigation of the Original Finding #11227

Open
farsheedify opened this issue Nov 9, 2024 · 18 comments
Labels

Comments

@farsheedify
Copy link
Contributor

farsheedify commented Nov 9, 2024

Bug description
In our organization, we are experiencing an issue with SAST scans where changes in code (the position of the vulnerable line) cause DefectDojo to mark the finding as mitigated and create a new finding for the same bug at the new line position. Currently, we are uploading the report with the scan type set to GitLab SAST Report in our CI/CD. As I understand, the default behavior for deduplication of static scans includes the line number in the algorithm.

I noticed that for Semgrep JSON Reports, DefectDojo uses the fingerprint ID provided by Semgrep for deduplication. I decided to test this change in the demo instance of DefectDojo to see if it could resolve our issue with SAST findings being closed and reopened due to line changes.

I scanned a sample repository on GitHub with Semgrep and generated a JSON report. I uploaded the report using the following curl command on the demo server:

curl -X "POST" "https://demo.defectdojo.org/api/v2/import-scan/" -H "accept: application/json" -H "Authorization: Token ***" -H "Content-Type: multipart/form-data" -F [email protected] -F "product_type_name=ptype_name" -F "active=true" -F "verified=true" -F "close_old_findings=true" -F "engagement_name=${GITHUB_RUN_ID}(SAST)" -F "build_id=${GITHUB_RUN_ID}" -F "minimum_severity=Info" -F "close_old_findings_product_scope=true" -F "scan_date=$TODAY" -F "engagement_end_date=$TODAY" -F "commit_hash=${GITHUB_SHA}" -F "product_name=${product_name}" -F "auto_create_context=true" -F "scan_type=Semgrep JSON Report" -F "tags=${GITHUB_REF_NAME}" -F "source_code_management_uri=${GITHUB_SERVER_URL}/${GITHUB_REPOSITORY}"

Originally, a bug was found on the 26th line of a file. After changing that line of code to the 21st line and rerunning the scan, the original bug was marked as mitigated, and the new bug was marked as a duplicate. Now, the bug has disappeared from the active bugs. The image below summarizes the situation. The bug and the file are the same. It seems that DefectDojo correctly marked the new finding as a duplicate of the previous finding (since only the line changed, but the fingerprints are the same). However, it incorrectly marked the original bug as mitigated, causing it to be removed from the active findings.

image

Steps to reproduce
Steps to reproduce the behavior:

  1. Upload a Semgrep JSON Report.
  2. Change the line of code for one of the findings.
  3. Rerun the scan.
  4. Upload the new generated report
  5. The finding with the changed line of code disappears from active findings.

Expected behavior
When rerunning the scan on the same file, if there is a change in the line of code, DefectDojo should:

1.Check the Fingerprint First: Use the fingerprint in the Semgrep report to identify findings.
2. Mark New Findings Correctly: Mark newly imported findings with similar instances (with different line numbers) as active and verified, since they can be assumed the latest instances.
3. Handle Original Findings Appropriately: Mark the original findings as duplicates (since they have outdated line of code).

Since the new finding contains the latest, updated line of code, it should be kept as the active finding. Ideally, DefectDojo should update the original finding's line of code without creating a new finding. This approach ensures that:

  1. If the number of duplicates to be kept is set to 0, there will be only one, updated instance per sast finding.
  2. We have the latest line of code for static findings.
@farsheedify farsheedify added the bug label Nov 9, 2024
@farsheedify
Copy link
Contributor Author

farsheedify commented Nov 26, 2024

I tested the same scenario, but this time I re-uploaded the same report using the UI (Ad Hoc Import), and the problem did not occur. It seems the issue only arises when importing the report through subsequent API imports, and possibaly related to engagement type, which are different (CI/CD vs Interactive). Here are the steps I followed:

  1. Scan the code and import the report via API: Initially, I scanned the code and imported the report using the API (CI/CD).
  2. Modify a finding and rescan: I changed the line number of one of the findings and rescanned the repository.
  3. Manual upload via UI: Finally, I manually uploaded the report through the UI (Interactive).

I plan to create an engagement with the type "Interactive" and then upload the test report into that engagement. This approach will be used instead of "auto creating context" in the import-scan request to determine if it resolves the issue.

Update:
I used "Interactive" as the engagement type. During the second import-scan, the finding was initially marked as mitigated but then reactivated immediately. As a result, the finding with the changed line of code remained open, although a comment about its auto-closure was added to the notes. However, when I performed a third import-scan, the finding was mitigated and subsequently disappeared from the active findings.

{DE8EFB3C-1F5B-4DEB-B7A0-FFBD2DC070A2}

{6CAB3EE4-7F7B-4E15-A95A-EA52B556A3CF}

@mtesauro
Copy link
Contributor

@farsheedify
So, a couple of things here:

  1. Engagement type isn't relevant. The only real difference between the two is the extra repo related meta-data you can add to a CICD engagement. Engagement type doesn't change how findings are handled besides grouping 1+ test results into something that can be reported on.
  2. You don't mention if you're doing imports or reimports - which you're doing is important. Imports are a one-time thing. Reimports require the scope of what's tested not to change and will automatically diff the current results with the previously ones.
  3. Updating finding status for things like dedup, etc is an async process so things changing is expected.
  4. It appears that line number is important in how you have deduplication configured. Look at the docs here. For Semgrep specifically, this is how dedup is handled

HTH

@farsheedify
Copy link
Contributor Author

farsheedify commented Nov 30, 2024

Thanks for your reply @mtesauro

  1. As you can see in the curl command I mentioned in the first post, the request is import-scan. Please also look for the flags in that command (e.g., close_old_findings_product_scope=true). Additionally, all subsequent imports are also done with import-scan. The screenshots and status of findings are reviewed after the async deduplication process.

  2. Additionally, since I use the Semgrep JSON Report, based on the documentation you provided, the deduplication is based on the unique id from the tool by default, which is the "fingerprint" parameter in the Semgrep report. Therefore, the line number should not affect the deduplication process, as we are not using the legacy deduplication algorithm that considers line numbers by default.

  3. The original finding that is getting mitigated has the same fingerprint as the duplicated one. This is clearly true, and DD correctly marks it as a duplicate. However, it incorrectly mitigates the original finding, leaving the duplicate finding as "inactive, duplicate". We expect the original finding to remain "active, verified" and not "inactive, verified, mitigated", while the duplicate one, with the different line of code, should be marked as "inactive, duplicate".

I can provide more accurate test data/scenario if it can help.

@mtesauro
Copy link
Contributor

mtesauro commented Dec 3, 2024

@farsheedify This is super interesting.

Is it possible you can add a sanitized scan file that shows this behavior so we can reproduce the issue?

@farsheedify
Copy link
Contributor Author

farsheedify commented Dec 4, 2024

@farsheedify This is super interesting.

Is it possible you can add a sanitized scan file that shows this behavior so we can reproduce the issue?

Sure.

I ran the scan on some sample files. To reproduce the issue, first upload semgrep-report_Line31.json using the following curl command:

curl -X "POST" "https://demo.defectdojo.org/api/v2/import-scan/" -H "accept: application/json" -H "Authorization: Token ${{ secrets.DEFECTDOJO_TOKEN }}" -H "Content-Type: multipart/form-data" -F [email protected] -F "product_type_name=clubpay" -F "active=true" -F "verified=true" -F "close_old_findings=true" -F "engagement_name=${GITHUB_RUN_ID}(SAST)" -F "build_id=${GITHUB_RUN_ID}" -F "minimum_severity=Info" -F "close_old_findings_product_scope=true" -F "scan_date=$TODAY" -F "engagement_end_date=$TODAY" -F "commit_hash=${GITHUB_SHA}" -F "product_name=${product_name}" -F "auto_create_context=true" -F "scan_type=Semgrep JSON Report" -F "tags=${GITHUB_REF_NAME}" -F "source_code_management_uri=${GITHUB_SERVER_URL}/${GITHUB_REPOSITORY}"

Next, upload semgrep-report_second_run_Line24.json using the same curl command. After the second upload, view all findings for the product. You will see two findings with the title:

python.lang.security.audit.dangerous-subprocess-use-tainted-env-args.dangerous-subprocess-use-tainted-env-args

The first finding (Original Finding) will be marked as "inactive, verified, mitigated" and is detected on sample/brute.py (Line 31). The second finding will be marked as "inactive, duplicate" and is detected on sample/brute.py (Line 24).
semgrep-report_Line31.json
semgrep-report_second_run_Line24.json

@mtesauro
Copy link
Contributor

mtesauro commented Dec 4, 2024

@farsheedify Thanks this helps a ton.

@Maffooch
Copy link
Contributor

I tested the scenario outlined above and have determined that close old findings and deduplication are working as expected, but are stomping on each other. Because close old findings on the import level waits for all findings to be created, and then compares the hash codes, deduplication is running when the finding is saved.

So on a finding that is expected to be a duplicate, deduplication will get to it first in the background via a celery worker, mark the new finding as a duplicate, and close it. Close old findings will get the saved finding back, but before deduplication changes its state, and is technically a stale record in this specific case. Close old findings will then close the original finding, and leave the new one open

These two smart features will then conflict with each other. Deduplication is saying "mark any new findings as mitigated because they are a duplicate of an existing finding" whereas close old findings says "mark older findings as mitigated because I want to focus on the newest finding found". This use case is essentially cancelling itself out, but leaving the state of the system in a weird spot with everything being closed.

A potential mitigation for this is to force the close old finding function (and therefore the entire import process) to wait on the async saving processes to finish, and then fetch the latest version of the same findings. Unfortunately, this will remove all performance gains of moving those processes to the background, for the sake of a very specific configuration that is not necessarily logical.

My advice here would be to assess your needs, and pick either close old findings or deduplication.

@mtesauro
Copy link
Contributor

@farsheedify Maffooch makes great observations in ☝️ I couldn't have written it up better.

@farsheedify
Copy link
Contributor Author

Thanks for your time and investigation.
I did not notice the impact of using the unique ID from the tool in the Semgrep report in the scenario you mentioned, as it seemed like the regular behavior of closing old findings and deduplicating the findings. This issue does not occur for other scan types, such as KICS Scan or Gitleaks Scan, and even when uploading the Semgrep report as a GitLab SAST Report. Moreover, I do not agree with Deduplication is saying "mark any new findings as mitigated because they are a duplicate of an existing finding. Normally when importing a report with duplciate findings, the new findings are marked as "inactive,duplicate", and not "inactive,duplicate,mitigated", except for this issue I reported.

Additionally, I tested this scenario using the "reimport" endpoint, and the issue did not occur. How is the reimport endpoint different from the import endpoint in terms of deduplication and closing old findings? Is it possible to have that behavior in the import endpoint as well? Although migrating to re-import endpoint from the import endpoint is super challenging for us due to this issue #3958.

I do not think this is a rare scenario; although it might go unnoticed for a long time. We had this issue for more than two years, but it came to our attention when we decided to export DefectDojo's data into a Power BI dashboard for data analysis. Our data analyst noticed that roughly more than 80 percent of our mitigated findings across 50 teams, with about 250 services, are related to SAST findings. After a deeper investigation, we found that most of these mitigations are due to changes in the position of the code (line number) and not actually the bug being fixed. This issue might be hidden for many clients. We decided to test switching to the "Semgrep JSON Report" instead of the "GitLab SAST Report" to migrate from "legacy" deduplication (which includes the line number in the hash) to the "unique ID from the tool" (which is the fingerprint of the Semgrep report for each finding). However, this issue is preventing us from switching our importing strategy.

I think the "calculation" of the hash code for deduplication is not applicable in this scenario as DefectDojo already has the unique ID from the tool (the fingerprint field in the JSON report of Semgrep).

From your clarifications, which I could be wrong about, it seems that the deduplication algorithm and closing old findings cannot be used together (which is not the case, as we are already using them together for all our test types from the start with no issues).

@valentijnscholten
Copy link
Member

valentijnscholten commented Jan 1, 2025

The code that marks the existing finding as mitigated (incorrectly) is in the import process. It looks like maybe something is going wrong there since the presence of the unique id should prevent stuff like this from happening. The hash codes for the existing finding and the "new" finding are different, so something in the hashcode calculation is going wrong.

image
image

When deduplication is disabled, the finding is still marked as mitigated. So I don't think deduplication is the cause of this.

@valentijnscholten
Copy link
Member

valentijnscholten commented Jan 1, 2025

Looks like the close_old_findings code doesn't take the DEDUPLICATION_ALGORITHM_PER_PARSER into account, and will purely look at the hash_code. So with or without deduplication enabled, this will never lead to the expected result.

I think the close_old_findings logic should be updated to respect the DEDUPLICATION_ALGORITHM_PER_PARSER setting.

What do you think @Maffooch @mtesauro

Additionally, I tested this scenario using the "reimport" endpoint, and the issue did not occur. How is the reimport endpoint different from the import endpoint in terms of deduplication and closing old findings? Is it possible to have that behavior in the import endpoint as well? Although migrating to re-import endpoint from the import endpoint is super challenging for us due to this issue #3958.

@farsheedify The problem in 3958 should not happen for Semgrep because of the unique id. Are you seeing this problem with Semgrep, or did you mention it because you saw it with other parsers?

@farsheedify
Copy link
Contributor Author

Thanks for your response @valentijnscholten.

The main issue with migrating from the import endpoint to the re-import endpoint is that the re-import endpoint operates on an engagement scope (except for the initial import). This means that if bugs in the same test type are not addressed during the first call to the re-import endpoint (which functions as an import), there is no way to close those bugs in the future, even if they are fixed. This situation causes significant trouble as we lose the "close old findings" functionality in 'product scope'.

Specifically for issue 3958, we have not tested the Semgrep JSON Report, as we have been uploading Semgrep reports using the GitLab SAST Report format, and we faced that issue with this test type, and other parsers. Currently we are using Anchore Grype, Trivy, KICS and Gitleaks other than Semgrep.

@farsheedify
Copy link
Contributor Author

I was wondering if this issue has been resolved in the latest version. If so, could you please share the related PR? @mtesauro

@paulOsinski
Copy link
Contributor

This means that if bugs in the same test type are not addressed during the first call to the re-import endpoint (which functions as an import), there is no way to close those bugs in the future, even if they are fixed. This situation causes significant trouble as we lose the "close old findings" functionality in 'product scope'.

@farsheedify can you explain what you mean here? If a new Finding is discovered by a 'reimport' it will simply be added to the existing Engagement as an Open Finding. Why would there be no way to close or act on a Finding discovered by /reimport in the future?

@valentijnscholten
Copy link
Member

valentijnscholten commented Jan 22, 2025

I still think the import process should respect the deduplication algorithm specified in the settings.

I do not understand this part though:

The main issue with migrating from the import endpoint to the re-import endpoint is that the re-import endpoint operates on an engagement scope (except for the initial import)

Reimport works within Test scope and it will close any findings that are no longer in the report that is being reimported. Because it's working in Test scope, it will not close down findings incorrectly and it will correctly use the unique id algorithm as configured in settings.

@farsheedify
Copy link
Contributor Author

farsheedify commented Jan 22, 2025

@paulOsinski @valentijnscholten Sure, I will explain what I meant by that, although it is not directly related to the issue being discussed, it is just the reason why we do not use the re-import endpoint.

From our tests last year, we used the re-import endpoint instead of the import endpoint to get the "delta" value and run notification scripts for newly imported findings, alerting our developers about the new findings in DefectDojo. During our tests, we encountered the following problem:

There were many open findings already imported using the import endpoint. Each time a new scan was run in the CI pipelines, we created a new engagement and imported the results into that engagement. Later, we decided to have a single engagement (let's call it Alpha engagement) and re-import scan results into this one engagement. Initially, most findings imported into the Alpha engagement were duplicates of previous findings. Over time, as some findings were mitigated, we observed that the status of duplicate findings in the Alpha engagement changed to "inactive, duplicate, mitigated." However, the original findings, which were previously imported into different engagements, remained in the "active, verified" status. I then noticed the following note in the API documentation:
Image
It means that during the first round of calling the re-import endpoint to upload results to the Alpha engagement (which is essentially an import), any mitigated findings would be closed as "inactive, verified, mitigated." However, after the first round, only the duplicate findings (that already exist in the Alpha engagement) would be marked as "mitigated," while the original findings would remain in the "active, verified" status, since the auto_close function in re-import does not work in product_scope. Our reports were Semgrep scan results, which we uploaded as GitLab SAST Reports to DefectDojo.

@valentijnscholten
Copy link
Member

valentijnscholten commented Jan 22, 2025

So there were still findings left outside the Alpha engagement but in other engagements in the same product?
I don't think that's a common scenario supported by reimport. But it might work if you set the Alpha engagement to perform deduplication inside that engagement only. It will prevent the duplicates and should allow reimport to close findings no longer present in uploaded reports. But you'll have to start with a new empty engagement, say Beta. Or you should remove the duplicate links from any findings that are marked as duplicate in Alpha (not sure if that's reliably possible).

Image

BTW The DD team is also looking the issue with import, so there may be some more information later, that's why I reopened the issue for now.

@farsheedify
Copy link
Contributor Author

farsheedify commented Jan 23, 2025

So there were still findings left outside the Alpha engagement but in other engagements in the same product?

Yes, exactly. We've been using the import feature for a long time. However, we wanted to take advantage of the "delta" in the response of the reimport. Having an event for "newly created findings" would be very useful in this case.

I don't think that's a common scenario supported by reimport.

You might be right; our problem is likely due to the migration process. However, I believe expecting the original finding to be mitigated when its duplicate is marked as mitigated is not unrelated. This might not occur if you only use the reimport endpoint for uploading reports for a product. I thought this issue might be related to #3958

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

No branches or pull requests

5 participants