Использование Content-Security-Policy вместе с React & Emotion
- вторник, 7 ноября 2023 г. в 00:00:19
Content-Security-Policy (CSP) - это HTTP заголовок, который улучшает безопасность веб-приложений за счет запрета небезопасных действий, таких как загрузка и отправка данных на произвольные домены, использование eval, inline-скриптов и т.д. В этой статье будет сделан фокус на директиве style-src и ее использование вместе с CSS-in-JS библиотекой emotion.
Content-Security-Policy заголовок должен быть выставлен в ответе вместе с загружаемой веб-страницей (например, index.html). Это выглядит следующим образом:
Content-Security-Policy: style-src 'self'
style-src - это директива, которая отвечает за то, какие стили можно загружать и применять на странице. Возможные значения:
'none' - все стили запрещены
'self' - разрешены файлы стилей, которые загружаются с того же домена, что и основной документ (страница)
<url> , например https://example.com - разрешены файлы стилей с этого домена, также допускаются wildcard (*) на месте под-домена и порта
'<hash-algorithm>-<base64-value>', например 'sha256-ozBpjL6dxO8fsS4u6fwG1dFDACYvpNxYeBA6tzR+FY8=' - разрешены файлы стилей и inline-стили (тег <style>), у которых хеш совпадает с указанным значением
'nonce-<value>' , например 'nonce-abc' - разрешаются inline-стили, у которых атрибут nonce совпадает с указанным (в примере - abc)
'unsafe-hashes' - разрешает inline-стили, указанные в атрибуте style строкой, например <div style="color:red;"></div>, при этом хеш значения атрибута должен совпадать с хешом, указанным в '<hash-algorithm>-<base64-value>'
'unsafe-inline' - разрешает все inline-стили, созданные через тег <style>
'unsafe-eval' - разрешает добавление/изменение CSS declarations, которые приводят к парсингу строки, например, с помощью CSSStyleDeclaration.cssText
Директива может принимать несколько значений через пробел. В этом случае это трактуется как логическое "или" - при удовлетворении хотя бы одному значению стили разрешаются.
emotion добавляет style элементы динамически и в последних версиях не может извлекать все стили в отдельный файл во время сборки приложения. Это означает, что для того, чтобы можно было использовать emotion вместе с style-src, есть следующие опции:
'unsafe-inline' - самая простая опция из всех. Не требует какой-либо настройки со стороны emotion. При этом мы снижаем безопасность нашего приложения, поэтому это решение можно использовать только как временное.
'nonce-<value>' - можно разрешить inline-стили, созданные emotion. Для этого нужно задать nonce при создании cache.
При использовании @emotion/react или @emotion/styled это можно сделать следующим образом:
import { CacheProvider } from "@emotion/react";
import createCache from "@emotion/cache";
export function App() {
const cache = createCache({
key: 'my-app',
nonce: getNonceValue(),
});
return (
<CacheProvider cache={cache}>
{/* children */}
</CacheProvider>
);
}Если используется @emotion/css напрямую, то потребуется создать свой экземпляр emotion:
import createEmotion from '@emotion/css/create-instance';
export const {
flush,
hydrate,
cx,
merge,
getRegisteredStyles,
injectGlobal,
keyframes,
css,
sheet,
cache
} = createEmotion({
key: 'my-app',
nonce: getNonceValue(),
});При использовании createEmotion потребуется поменять все места, где раньше импортировался @emotion/css на этот модуль:
// import { css } from "@emotion/css";
import { css } from "./emotion";Т.к. значение CSP заголовка недоступно коду, исполняемому на клиенте, то значение нужно дополнительно передать другим образом. Один из вариантов - это создание inline-скрипта со значением, которое выставляется на бекенде:
<script id="nonce" type="application/json">
"abc"
</script>На фронтенде это можно использовать таким образом:
function getNonceValue() {
const nonceElement = document.getElementById("nonce");
return JSON.parse(nonceElement.textContent);
}Обратите внимание на type="application/json" - таким образом браузер не считает это исполняемым кодом, и особое значение для script-src не требуется.