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

feat:created README.md file for best security practices react #69

Merged
165 changes: 165 additions & 0 deletions examples/react-security-best-practices/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,165 @@
# React security best practices ✨

Welcome to our guide on improving security of you react web application 🚀

Looking for the best ways to secure your React app? Then you’ve come to the right place! We’ve created this checklist of React security best practices to help you and your team find and fix security issues in your React applications. We’ll also show you how to automatically test your React code for security-related issues and automatically fix them. Let's dive in and discover what they are and how they can secure your website!

## Use Default XSS Protection with Data Binding 🚀

### Firstly what is XSS?

XSS, or Cross-Site Scripting, is a type of security vulnerability that can occur in any web applications. XSS happens when an application allows untrusted data to be included in web pages that are then viewed by other users. This can lead to malicious code execution in the context of a user's browser, potentially compromising the security and privacy of that user.

So basically it happens when we render user generated data directly in DOM without proper sanitization.

### How to prevent XSS attacks?

Use default data binding with curly braces `` {} `` and React will automatically escape values to protect against XSS attacks. Note that this protection only occurs when rendering textContent and not when rendering HTML attributes.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The statement "React will automatically escape values to protect against XSS attacks" is partially correct. While React does escape values when rendering text content, it doesn't do so for HTML attributes or JSX properties. This could lead to potential security vulnerabilities if not handled correctly. Consider clarifying this point in your guide.


For example:-

`` <div>{data}</div> ``

**Limitations of React's Default XSS Protection:**

While React's default behavior provides a level of XSS protection by escaping content when rendering, it's important to note that it may not cover all cases:

1. **dangerouslySetInnerHTML**: React provides a mechanism called `dangerouslySetInnerHTML`, which allows you to inject raw HTML into a component. This feature intentionally bypasses React's automatic escaping and should be used with extreme caution. Developers are responsible for ensuring that the HTML content provided through `dangerouslySetInnerHTML` is safe and properly sanitized.

2. **Dynamic Attribute Values**: React's default data binding doesn't automatically sanitize attribute values that are set dynamically. If you're dynamically setting attributes like `src`, `href`, or event handlers, you need to ensure that the values are properly sanitized to prevent potential XSS vulnerabilities.

## Watch Out for Dangerous URLs and URL-Based Script Injection

URLs can contain dynamic script content via javascript: protocol URLs. Use validation to assure your links are http: or https: to avoid javascript: URL-based script injection. Achieve URL validation using a native URL parsing function then match the parsed protocol property to an allow list.

For example:-

```Javascript
function validateURL(url) {

const parsed = new URL(url)

return ['https:', 'http:'].includes(parsed.protocol)

}

<a href={validateURL(url) ? url : '#'}>Click here!</a>

```
Comment on lines +38 to +48
Copy link
Contributor

@coderabbitai coderabbitai bot Oct 3, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The function validateURL(url) is a good example of how to validate URLs to prevent script injection. However, it would be better to throw an error or return a default safe URL when the input URL is invalid, rather than just returning a boolean. This way, the function can be used directly in the href attribute without additional checks.

- function validateURL(url) {
-     const parsed = new URL(url)
-     return ['https:', 'http:'].includes(parsed.protocol)
- }
- <a href={validateURL(url) ? url : '#'}>Click here!</a>
+ function validateURL(url) {
+     try {
+         const parsed = new URL(url)
+         if (['https:', 'http:'].includes(parsed.protocol)) {
+             return url;
+         } else {
+             throw new Error('Invalid URL protocol');
+         }
+     } catch (error) {
+         console.error(error);
+         return '#'; // Default safe URL
+     }
+ }
+ <a href={validateURL(url)}>Click here!</a>

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@ulises-jeremias Understood, I will disregard the previous comment. If you have any other requests or need further assistance, feel free to ask.


The above code will prevent javascript files to be injected.

Don’t use this:- `` <a href={attackerControlled}>Click here!</a> ``

One more important thing to add is we need to validate any user input data on both client side + server side to prevent any tampering attempts made by attacker.

## Sanitize and Render HTML

One of the ways to do XSS attacks is by possbily inserting HTML directly into rendered DOM nodes using dangerouslySetInnerHTML. Any content inserted this way must be sanitized beforehand. Use a sanitization library like **dompurify** on any values before placing them into the dangerouslySetInnerHTML prop.

Use **dompurify** when inserting HTML into the DOM.

For example:-

```Javascript
import purify from "dompurify";

<div dangerouslySetInnerHTML={{ __html:purify.sanitize(data) }} />

```

But the developer should still note that even after sanitization, `dangerouslySetInnerHTML` should be used sparingly and only when absolutely necessary, as it increases the risk of XSS attacks.

## Avoid Direct DOM Access

Always avoid accessing the DOM to inject content into DOM nodes directly. But if you do have to, use `dangerouslySetInnerHTML` to inject HTML and sanitize it before injecting it using **dompurify**.

**Why Direct DOM Access is Discouraged in React:**

1. **Breaks the Unidirectional Data Flow**: React follows a unidirectional data flow, which means that data flows in one direction—from parent components to child components. Directly manipulating the DOM breaks this flow by introducing an external source of truth. When you bypass React's virtual DOM and directly manipulate the DOM, React loses control over the component's state and can't accurately track changes. This can lead to unexpected behavior and make your application harder to reason about and maintain.

2. **Inefficient and Error-Prone**: React is designed to efficiently update the DOM by using a virtual DOM and a reconciliation process that minimizes actual DOM manipulations. When you directly manipulate the DOM, you risk introducing inefficiencies, such as unnecessary re-renders, and you may encounter difficult-to-debug.

3. **Security Concerns**: Direct DOM access can also introduce security vulnerabilities, such as cross-site scripting (XSS). By bypassing React's mechanisms for safely rendering user-generated content, you expose your application to potential security risks.

4. **Maintenance Challenges**: Code that relies on direct DOM access can become harder to maintain over time, especially in larger applications or when multiple developers are involved. React's component-based architecture and declarative approach make it easier to understand and maintain your codebase.

### What is react `dangerouslySetInnerHTML` ?

`dangerouslySetInnerHTML` is a property that enables developers to directly insert html content within an HTML element found in a React app.

```Javascript

import purify from "dompurify";

<div dangerouslySetInnerHTML={{__html:purify.sanitize(data) }} />

```

## Secure React server-side rendering

Data binding will provide automatic content escaping when using server-side rendering functions like `ReactDOMServer.renderToString()` and `ReactDOMServer.renderToStaticMarkup()`.

Avoid concatenating strings onto the output of `renderToStaticMarkup()` before sending the strings to the client for hydration.

To avoid XSS, don’t concatenate unsanitized data with the output of `renderToStaticMarkup()`

**Why Concatenating Unsanitized Data with `renderToStaticMarkup()` Output is Risky for XSS Attacks:**

1. **Injection of Malicious Code**: If you concatenate unsanitized or untrusted data directly into the HTML generated by `renderToStaticMarkup()`, you open up the possibility of injecting malicious code, such as JavaScript, into your rendered HTML. Attackers can craft payloads that include script tags or other HTML elements, and if these payloads are included in your page, they can execute arbitrary code in the context of your web application.

2. **XSS Attack Surface**: By not properly sanitizing or escaping user-generated data, you create an XSS attack surface. Attackers can exploit this surface to inject scripts that steal user data, perform actions on behalf of users, or otherwise compromise the security and integrity of your application.

3. **Browser Security Context**: When a browser encounters unsanitized data in your HTML, it interprets it as code to execute, not as plain text. This behavior is what makes XSS attacks possible. Properly escaping or sanitizing data ensures that it is treated as data and not executable code.

## Check for Known Vulnerabilities in Dependencies

Some versions of third-party components might contain JavaScript security issues. Always check your dependencies with a software composition analysis (SCA) tool before adding them to a project, and be sure to update when a newer version becomes available. The following ones are recommended to check for vulnerabilities

- [Snyk](https://snyk.io/)

- [Github dependabot](https://github.com/dependabot)

- [Sonar Qube](https://www.sonarsource.com/products/sonarqube/)

## Avoid JSON Injection Attacks

It is common to send JSON data along with server-side rendered React pages. Always escape `<` characters with a benign value to avoid injection attacks.

Always escape HTML significant values from JSON with benign equivalent characters.

For example:

`` window.__PRELOADED_STATE__ = ${JSON.stringify(preloadedState).replace( /</g, '\\u003c')} ``

Here replacing the `<` character with `\u003c` is a security measure to ensure that potentially malicious data included in JSON does not have unintended consequences when rendered in HTML context. It's a part of a broader strategy to protect against JSON injection and XSS attacks by properly escaping and validating data.

## Use non-vulnerable versions of React

The React library has had a few [high severity vulnerabilities](https://snyk.io/vuln/npm:react/) in the past, so it is a good idea to stay up to date with the latest version.

Avoid vulnerable versions of the `react` and `react-dom` by verifying that you are on the latest version using `npm outdated` to see the latest versions.See how to use this command below

Navigate to your code directory first and then run

`npm outdated`

It will show all the outdated packages in a tabular format.

## Use linter configurations

Install linter configurations and plugins that will automatically detect security issues in your code and offer remediation advice.

Use the [ESLint React security config](https://github.com/snyk-labs/eslint-config-react-security/) to detect security issues in our code base.

Configure a pre-commit hook that fails when security-related linter issues are detected using a library like [husky](https://github.com/typicode/husky).

## Avoid dangerous library code

Library code is often used to perform dangerous operations like directly inserting HTML into the DOM. Review library code manually or with linters to detect unsafe usage of React’s security mechanisms.

Consider the use of libraries with a caution that do use `dangerouslySetInnerHTML`, `innerHTML`, unvalidated URLs or other unsafe patterns. Use security linters on your `node_modules` folder to detect unsafe patterns in your library code.

## Conclusion 🌟

Security is a very important for any web app out there that uses React. By implementing the recommendations you can protect your website from all types of external scripts and attacks.