OWASP Top 10 Risks for Open Source Software

top 10 oss risks

Introduction

Despite the heavy reliance on OSS in the software supply chain, the industry lacks a consistent way to understand and measure risk for OSS. Risk management in OSS started with license management, and then evolved to CVEs, but we still lack a holistic approach to OSS risk management that encompasses security, legal, and operational aspects. With this document, we’re excited to collaborate with industry experts and leaders to create just that.

Over the last decade of reliance on OSS, known vulnerabilities, captured as CVEs, have emerged as the key metric of security. Known vulnerabilities, while an important signal, typically capture mistakes made by well-intentioned developers. These mistakes could be exploited by attackers and should be fixed, but they hardly encompass the full spectrum of risks that a reliance on OSS includes.

Operational risks, like ones introduced by outdated or unmaintained software, or next-generation supply chain attacks like name confusion attacks, cannot be captured by CVEs. These risks are significant, as highlighted by the recent Open Source Security and Risk Analysis report by Synopsys:

  • 89% of codebases contain OSS that is more than 4 years out of date
  • 91% of codebases contain components that have had no new development in over two years

On The State of Dependency Management, the Station 9 research team from Endor Labs uncovered that 95% of vulnerabilities exist in transitive dependencies (the software packages automatically brought in by the OSS selected by developers). And out of those, many are not actually reachable, or will cause a devastating ripple effect of incompatibility if they were updated. With this list, which was peer-reviewed and contributed to by over 20 CISOs and CTOs, the team sought to find the top risks security and development teams should be ready for, both operational and security.

The top 10 OSS risks are:

  • OSS-RISK-1 Known Vulnerabilities: A component version may contain vulnerable code, accidentally introduced by its developers. Vulnerability details are publicly disclosed, e.g, through CVE, GitHub Security Advisories or other, more informal communication channels. Exploits and patches may or may not be available.
  • OSS-RISK-2 Compromise of Legitimate Package: Attackers may compromise resources that are part of an existing legitimate project or of the distribution infrastructure in order to inject malicious code into a component, e.g, through hijacking the accounts of legitimate project maintainers or exploiting vulnerabilities in package repositories.
  • OSS-RISK-3 Name Confusion Attacks: Attackers may create components whose names resemble names of legitimate open-source or system components (typo-squatting), suggest trustworthy authors (brand-jacking) or play with common naming patterns in different languages or ecosystems (combo-squatting).
  • OSS-RISK-4 Unmaintained Software: A component or component version may not be actively developed any more, thus, patches for functional and non-functional bugs may not be provided in a timely fashion (or not at all) by the original open source project.
  • OSS-RISK-5 Outdated Software: A project may use an old, outdated version of the component (though newer versions exist).
  • OSS-RISK-6 Untracked Dependencies: Project developers may not be aware of a dependency on a component at all, e.g., because it is not part of an upstream component’s SBOM, because SCA tools are not run or do not detect it, or because the dependency is not established using a package manager.
  • OSS-RISK-7 License Risk: A component or project may not have a license at all, or one that is incompatible with the intended use or whose requirements are not or cannot be met.
  • OSS-RISK-8 Immature Software: An open source project may not apply development best-practices, e.g., not use a standard versioning scheme, have no regression test suite, review guidelines or documentation. As a result, a component may not work reliably or securely.
  • OSS-RISK-9 Unapproved Change: A component may change without developers being able to notice, review or approve such changes, e.g., because the download link points to an unversioned resource, because a versioned resource has been modified or tampered with or due to an insecure data transfer.
  • OSS-RISK-10 Under/over-sized Dependency: A component may provide very little functionality (e.g. npm micro packages) or a lot of functionality (of which only a fraction may be used).

Authors and Contributors

Authors

Dependency Management 101

To better understand how these risks may affect us, let us quickly review some basic concepts of dependency management using a simple example. Just skip this section if you’re familiar with dependency management.

The root node of the dependency graph displayed below represents the example project. The child nodes of the root represent 9 open source components the project “directly” depends on. Those components, however, depend on other components as well, all of which become “transitive” or “indirect” dependencies from the perspective of the top-level project.

Direct dependencies are consciously selected by the project developers, e.g., through the declaration in a manifest file such as package.json (Node.js/npm) or pom.xml (Java/Maven) file. Package managers take care of downloading and installing those direct dependencies from 3rd party package repositories to the developers workstation or a CI/CD system. When doing so, package managers also identify transitive dependencies, resolve potential version conflicts and install them locally. In other words, plenty of components are downloaded in an automated fashion in order to make sure all code required by the example project (and more) is present on the developer machine.

Dependency Graph for Graph Maven Plugin

This high degree of automation boosted the reuse of open source components in today’s software development. As a result, large portions of modern software have not been specifically developed for an application at hand, but come from generic open source projects. Looking again at the dependency graph, it is not surprising that the ratio of code written for the project itself (the root node) to the code of open source components is 1:4 or less. And we haven’t even touched upon IDEs, build tools etc.


OSS-RISK-1 : Known vulnerabilities in dependencies

Description:

A component version may contain vulnerable code, accidentally introduced by its developers. Vulnerability details are publicly disclosed, e.g, through CVE, GitHub Security Advisories or other, more informal communication channels. Exploits and patches may or may not be available.

The vulnerability may be exploitable in the context of the downstream software, which could compromise the confidentiality, integrity or availability of the respective system or its data, allow lateral movements in the target environment or have other negative effects.

Examples:

  1. CVE-2017-5638 in Apache Struts, which caused the Equifax data breach
  2. CVE-2021-44228 in Apache Log4j, also called Log4Shell

Actions:

  1. Monitor applications, containers and systems for the presence of (direct and transitive) open source dependencies with known vulnerabilities
  2. Prioritize the analysis and mitigation on the basis of, for instance
    • Scores and metrics like CVSS and EPSS
    • Threat intelligence like the availability and maturity of exploit code, or information on attacks observed in the wild such as provided by the CISA KEV catalog
    • DAST and SAST tools, which determine if vulnerable code can be executed in the context of the dependent software

References:

  1. OWASP Top 10:2021 A06:2021 - Vulnerable and Outdated Components

OSS-RISK-2 : Compromise of Legitimate Package

Description:

Attackers may compromise resources that are part of an existing legitimate project or of the distribution infrastructure in order to inject malicious code into a component, e.g, through hijacking the accounts of legitimate project maintainers or exploiting vulnerabilities in package repositories.

Malicious code can be executed on end-user systems or on systems belonging to the organization that develops and/or operates the dependent software (e.g., build systems or developer workstations). The confidentiality, integrity and availability of systems and the data processed/stored thereon is at risk.

Examples:

  1. Event-stream: This attack on a legitimate component targeted users of Copay Bitcoin wallets.
  2. The SolarWinds Cyber-Attack

Actions:

There’s no single action to detect and prevent the ingestion of compromised packages. Organizations should consult emerging standards and frameworks like the Secure Supply Chain Consumption Framework (S2C2F) to inform themselves about possible safeguards, which should be selected and prioritized according to individual security requirements and risk appetite.

Example actions comprise:

  1. Verify component provenance according to SLSA
  2. Build component from the sources (yourself or a trusted 3rd-party)
  3. Review code manually and/or automatically
  4. Retrieve all components from a secured internal store (such binary repositories host home-made binaries and mirror external components)

References:

  1. Secure Supply Chain Consumption Framework (S2C2F)
  2. Risk Explorer for Software Supply Chains Subvert Legitimate Package (AV-001)
  3. Supply chain Levels for Software Artifacts (SLSA)
  4. MITRE ATT&CK T1195.001 Compromise Software Dependencies and Development Tools
  5. OWASP Top 10 CI-CD Security Risks CICD-SEC-3: Dependency Chain Abuse
  6. OWASP Software Component Versification Standard (SCVS) V6 Pedigree and Provenance

OSS-RISK-3 : Name Confusion Attacks

Description:

Attackers may create components whose names resemble names of legitimate open-source or system components (typo-squatting), suggest trustworthy authors (brand-jacking) or play with common naming patterns in different languages or ecosystems.

Malicious code can be executed on end-user systems or on systems belonging to the organization that develops and/or operates the dependent software (e.g., build systems or developer workstations). The confidentiality, integrity and availability of systems and the data processed/stored thereon is at risk.

Examples:

  1. Colourama (PyPI, 2018): Typo-squatting attack on the legitimate Python package “colorama”, redirecting Bitcoin transfers to an attacker-controlled wallet.

Actions:

Prior to installing/using a component:

  1. Check code characteristics (pre/post installation hooks, encoded payloads, etc.) and project characteristics (source code repository, maintainer accounts, release frequency, number of downstream users, etc.) for leading risk indicators.

    Note that some component metadata is not verified by package repositories, thus, can easily be forged by attackers.

  2. Verify that the component carries a signature from a trusted party (for ecosystems that support/require signatures)

References:

  1. Secure Supply Chain Consumption Framework (S2C2F)
  2. Risk Explorer for Software Supply Chains Create Name Confusion with Legitimate Package (AV-200)
  3. Supply chain Levels for Software Artifacts (SLSA)
  4. MITRE ATT&CK T1195.001 Compromise Software Dependencies and Development Tools

OSS-RISK-4 : Unmaintained software

Description:

A component or component version may not be actively developed or supported any more, i.e. patches for new functional and security bugs may not be developed.

Patch development may therefore need to be done by downstream developers with potentially less experience and knowledge regarding the affected component.

This can result in increased efforts and longer resolution times. During that time, system access or functionality may need to be restricted to avoid continued exposure.

Examples:

  1. core-js (npm, 2020)
  2. Gorilla Web Toolkit (Go, 2022)
  3. minimist (npm, 2022)

Actions:

  1. Check indicators for project liveliness and health.

    However, note that little activity can also be a sign of maturity. Projects that are considered feature-complete and mature will see less activity than projects under active development, and still receive timely patches in case of problems.

    Example indicators:

    • Recent issue and commit activity means the project is active.
    • A high ratio of issues opened by external contributors indicates that the project is active.
    • Activity from corporate affiliated accounts that indicate that the project can have reliable backing and support.
    • Activity from reputable accounts indicates that the repository is well-maintained.
    • The repository has frequent releases indicating a commitment to maintaining and supporting the codebase.
  2. Search for information on a project’s maintenance or support strategy, e.g., the presence and dates of long-term support (LTS) versions.

    The Spring project is a great example for documenting support time frames.

  3. Check the project page to see whether its has been archived, or whether there are any explicit statements regarding the project’s maintenance status.

References:

  1. OWASP Top 10:2021 A06:2021 - Vulnerable and Outdated Components
  2. Bus factor
  3. Common Weakness Enumeration (CWE) CWE-1104: Use of Unmaintained Third Party Components
  4. CWE-1329: Reliance on Component That is Not Updateable
  5. Adoptoposs
  6. Jazzband

OSS-RISK-5 : Outdated software

Description:

A project may use an old, outdated version of the component (though newer versions exist).

Falling too much behind the latest releases of a dependency can make it difficult to perform timely updates in emergency situations, e.g., when a vulnerability is disclosed for the version in use. Old releases may also not receive the same level of security assessment as recent versions, esp. whether they are affected by vulnerabilities. If a new version is syntactically or semantically incompatible with the current version in use, application developers may require significant update/migration efforts to resolve the incompatibility.

Examples:

  1. Spring Boot support time frames
  2. Vulnerabilities affecting Apache Tomcat are “not checked against the 8.0.x branch and will not be fixed”

Actions:

  1. Make dependency updates recurring backlog items
  2. Automate the discovery and suggestion of updates, e.g. through merge/pull requests
  3. Detect the introduction or breaking changes through change impact analysis tools

References:

  1. OWASP Top 10:2021 A06:2021 - Vulnerable and Outdated Components
  2. Tools to check for the availability of new versions:
  3. Tools to detect breaking changes:

OSS-RISK-6 : Untracked Dependencies

Description:

Project developers may not be aware of a dependency on a component at all, e.g., because it is not part of an upstream component’s SBOM, because SCA tools are not run or do not detect it, or because the dependency is not established using a package manager.

Flying under the radar, the respective component cannot be checked or monitored for any of the other deficiencies.

Examples:

  1. Incomplete SBOMs received for upstream components or produced by SCA tools
  2. Inclusion of 3rd-party code in a managed (tracked) dependency, e.g.
    • code snippets
    • source code files (copied as-is into the dependency’s sources, also called “vendored”)
    • compiled code (e.g., platform-specific binaries or Java archives/class files, also called rebundling)
  3. Dependencies not established through the manifest files of package managers like PIP or Maven, e.g. manual or scripted installation through brew or apt-get
  4. IDE plugins, build scripts, test dependencies or other developer tools, though not included in the dependent software itself, still pose security and operational risks

Actions:

  1. Evaluate and compare SCA tools regarding their capability to produce accurate bills of materials, both at coarse-granular level (e.g., dependencies declared with help of package management tools likes Maven or npm) and fine-granular level (e.g., artifacts like single files included “out of band”, i.e., without using package managers).

References:

  1. OWASP Software Component Versification Standard (SCVS) V1 Inventory and V2 Software Bills of Materials
  2. Research on rebundling
  3. Anand Sawant: Dependency Resolution in Python: Beware The Phantom Dependency (2023)

OSS-RISK-7 : License and Regulatory Risk

Description:

A component or project may not have a license at all, one that is incompatible with the intended use by a downstream consumer, or one whose requirements are not or cannot be met by a downstream user.

A component may also violate license terms independent from downstream use, e.g., if it is licensed as GPL but includes files licensed under the original (4-clause) BSD license.

A component may also conflict with legal and regulatory requirements, e.g., related to FedRAMP certification or export control.

It is important to use components in compliance with their license terms. The absence of a license or non-compliant use can result in copyright or license infringements, which the copyright holder can take legal action against.

The violation of legal and regulatory requirements can constrain or hamper addressing certain verticals or markets.

Examples:

  1. Free Software Foundation, Inc. v. Cisco Systems, Inc. (2008)

Actions:

  1. Identify acceptable licenses for the intended use of the component in the software under development.

    This should consider, for example, how the component is linked, the software’s deployment model (cloud, on-premise/device) and the intended distribution scheme.

  2. Comply with requirements stated in the open source licenses.
  3. Avoid components without license.
  4. Scrutinize component files for multiple and/or incompatible licenses.

References:

  1. OSI Approved Licenses
  2. SPDX License List
  3. REUSE
  4. Various Licenses and Comments about Them
  5. Fossology

OSS-RISK-8 : Immature-Software

Description:

An open source project may not apply development best-practices, e.g., not use a standard versioning scheme, have no regression test suite, no development or review guidelines or no documentation. As a result, a component may not work reliably or securely (in the sense of having security weaknesses that result in exploitable vulnerabilities).

The dependency on an immature component or project comes with operational risks. The dependent software may not work as expected and result in runtime reliability issues, or its use may be overly complex and expensive for the dependent software development organization.

For example, a component or project may lack documentation, may not use or comply with an established versioning scheme (which can result in breaking changes during component updates), or may not have a test suite to discover regressions introduced through pull/merge requests. Such cases can increase the effort of developers depending on such components.

Examples:

  • None

Actions:

  1. Check quality indicators and whether a project follows development best-practices.

    Example indicators:

    • The project includes test code.
    • Displaying the Code Coverage badge means that the repository is using code coverage tools in its development process.
    • The repository includes documentation making it easier to understand and use.
    • The repository uses CI and a high fraction of commits pass the CI checks which is a sign of good code quality.
    • When a repository contains binary files it is harder to analyze and assess its functionality and risks.
  2. A proxy for checking project maturity may also be the number of downstream dependents.

References:

  1. OpenSSF Best Practices Badge Program

OSS-RISK-9 : Unapproved Change (Mutable)

Description:

A component may change without developers being able to notice, review or approve such changes, e.g., because the download link points to an unversioned resource, because a versioned resource has been modified or tampered with or due to an insecure data transfer.

Using components that are not guaranteed to be identical when downloaded at different points in time are primarily a security risk. Attacks such as on the Codecov Bash Uploader demonstrate the risk of piping downloaded scripts directly to bash, without checking their integrity beforehand. Mutable components also threaten the stability and reproducibility of software builds.

Examples:

  1. References to non-versioned shell scripts in CI/CD pipelines
  2. References to Git repositories without commit identifier
  3. Insecure HTTP links to package registries

Actions:

  1. Use resource identifiers providing guarantees (or at least some degree of assurance) to always point to the same, immutable artifact.
  2. Verify digests or signatures after component download and before installation/use
  3. Use secure protocols for connection/distribution to avoid MITM attacks

References:

  1. SLSA Immutable Reference
  2. CWE-829: Inclusion of Functionality from Untrusted Control Sphere
  3. CWE-830: Inclusion of Web Functionality from an Untrusted Source
  4. OWASP Top 10:2021 A08:2021 - Software and Data Integrity Failures

OSS-RISK-10 : Under/over-sized Dependency

Description:

A component may provide very little functionality (e.g. npm micro packages) or a lot of functionality (of which only a fraction may be used).

Very small components, e.g. ones containing few lines of code only, are subject to the same supply chain risks as large ones, e.g. account take-over, malicious pull requests or CI/CD vulnerabilities, for comparably little functionality. In other words, in exchange for very few lines of code used, the consumer’s security becomes dependent on the upstream project’s security and development posture.

Very large components, on the other hand, may have accumulated many features that are not needed in standard use-cases, but contribute to the component’s attack surface. Additionally, such unused features may also bring in additional, unused dependencies (bloated dependencies).

Examples:

  1. Apache Log4j

    Included functionality to download and execute arbitrary Java classes from remote servers, which eventually led to Log4Shell.

  2. Left-pad (npm, 2016)

    Contained 11 lines of code to pad strings. Its removal from npm broke the builds of numerous downstream consumers.

Actions:

  1. Become aware of unused component capabilities, esp. if they use critical (security sensitive) APIs such as to establish network connections.

    Evaluate possibilities to disable unused capabilities, or move to smaller alternative open source components with fewer capabilities.

  2. Become aware of micro packages, and consider redeveloping their functionality internally.

References:

  1. Definition: Feature creep
  2. Tools to uncover the use of security sensitive APIs:
    • Google Capslock for Go
    • Microsoft Application Inspector for various programming languages
    • Rahman et al.: Less Is More: A Mixed-Methods Study on Security-Sensitive API Calls in Java for Better Dependency Selection