mirror of
https://github.com/QuivrHQ/quivr.git
synced 2024-12-15 01:21:48 +03:00
d0e363e66e
# Description Enable CSP in all environments (local/preview/prod). Relies on NEXT_PUBLIC_ENV env variable, which should be `'local'|'preview'|'prod'` # Comparison of old and new CSP values (tested locally) ## Before ### CSP (for prod only) ``` default-src 'self' https://fonts.googleapis.com https://xxx.supabase.co https://api.june.so https://www.quivr.app/; connect-src 'self' https://xxx.supabase.co http://localhost:5050 https://api.june.so https://api.openai.com https://cdn.growthbook.io https://vitals.vercel-insights.com/v1/vitals; img-src 'self' https://www.gravatar.com data:; media-src 'self' https://user-images.githubusercontent.com https://www.quivr.app/ https://quivr-cms.s3.eu-west-3.amazonaws.com; script-src 'unsafe-inline' 'unsafe-eval' https://va.vercel-scripts.com/ https://www.quivr.app/ https://www.google-analytics.com/; frame-ancestors 'none'; style-src 'unsafe-inline' https://www.quivr.app/; ``` ## After ### Prod CSP (iso with before) ``` default-src 'self' https://fonts.googleapis.com https://xxx.supabase.co https://api.june.so https://www.quivr.app/; connect-src 'self' https://xxx.supabase.co http://localhost:5050 https://api.june.so https://api.openai.com https://cdn.growthbook.io https://vitals.vercel-insights.com/v1/vitals; img-src 'self' https://www.gravatar.com data:; media-src 'self' https://user-images.githubusercontent.com https://www.quivr.app/ https://quivr-cms.s3.eu-west-3.amazonaws.com; script-src 'unsafe-inline' 'unsafe-eval' https://va.vercel-scripts.com/ https://www.quivr.app/ https://www.google-analytics.com/; frame-ancestors 'none'; style-src 'unsafe-inline' https://www.quivr.app/; ``` ### Preview CSP ``` default-src 'self' https://fonts.googleapis.com https://xxx.supabase.co https://api.june.so https://preview.quivr.app/; connect-src 'self' https://xxx.supabase.co http://localhost:5050 https://api.june.so https://api.openai.com https://cdn.growthbook.io https://vitals.vercel-insights.com/v1/vitals; img-src 'self' https://www.gravatar.com data:; media-src 'self' https://user-images.githubusercontent.com https://www.quivr.app/ https://quivr-cms.s3.eu-west-3.amazonaws.com; script-src 'unsafe-inline' 'unsafe-eval' https://va.vercel-scripts.com/ https://preview.quivr.app/ https://www.google-analytics.com/; frame-ancestors 'none'; style-src 'unsafe-inline' https://preview.quivr.app/; ``` ### Local CSP ``` default-src 'self' https://fonts.googleapis.com https://xxx.supabase.co https://api.june.so http://localhost:3000 http://localhost:3001; connect-src 'self' https://xxx.supabase.co http://localhost:5050 https://api.june.so https://api.openai.com https://cdn.growthbook.io https://vitals.vercel-insights.com/v1/vitals; img-src 'self' https://www.gravatar.com data:; media-src 'self' https://user-images.githubusercontent.com https://www.quivr.app/ https://quivr-cms.s3.eu-west-3.amazonaws.com; script-src 'unsafe-inline' 'unsafe-eval' https://va.vercel-scripts.com/ http://localhost:3000 http://localhost:3001 https://www.google-analytics.com/; frame-ancestors 'none'; style-src 'unsafe-inline' http://localhost:3000 http://localhost:3001; ``` # 🧪 External checks Syntax checked with https://csp-evaluator.withgoogle.com/ (for the 3 environments). --------- Co-authored-by: gozineb <zinebe@theodo.fr>
163 lines
4.2 KiB
JavaScript
163 lines
4.2 KiB
JavaScript
/* eslint-disable max-lines */
|
|
const nextConfig = {
|
|
images: {
|
|
domains: [
|
|
"www.quivr.app",
|
|
"quivr-cms.s3.eu-west-3.amazonaws.com",
|
|
"www.gravatar.com",
|
|
],
|
|
},
|
|
// eslint-disable-next-line prefer-arrow/prefer-arrow-functions
|
|
async headers() {
|
|
return [
|
|
{
|
|
source: "/(.*)",
|
|
headers: securityHeaders,
|
|
},
|
|
];
|
|
},
|
|
};
|
|
|
|
const ContentSecurityPolicy = {
|
|
"default-src": [
|
|
"'self'",
|
|
"https://fonts.googleapis.com",
|
|
process.env.NEXT_PUBLIC_SUPABASE_URL,
|
|
"https://api.june.so",
|
|
{
|
|
prod: "https://www.quivr.app/",
|
|
preview: "https://preview.quivr.app/",
|
|
local: ["http://localhost:3000", "http://localhost:3001"],
|
|
},
|
|
],
|
|
"connect-src": [
|
|
"'self'",
|
|
process.env.NEXT_PUBLIC_SUPABASE_URL,
|
|
process.env.NEXT_PUBLIC_BACKEND_URL,
|
|
"https://api.june.so",
|
|
"https://api.openai.com",
|
|
"https://cdn.growthbook.io",
|
|
"https://vitals.vercel-insights.com/v1/vitals",
|
|
],
|
|
"img-src": ["'self'", "https://www.gravatar.com", "data:"],
|
|
"media-src": [
|
|
"'self'",
|
|
"https://user-images.githubusercontent.com",
|
|
"https://www.quivr.app/",
|
|
"https://quivr-cms.s3.eu-west-3.amazonaws.com",
|
|
],
|
|
"script-src": [
|
|
"'unsafe-inline'",
|
|
"'unsafe-eval'",
|
|
"https://va.vercel-scripts.com/",
|
|
{
|
|
prod: "https://www.quivr.app/",
|
|
preview: "https://preview.quivr.app/",
|
|
local: ["http://localhost:3000", "http://localhost:3001"],
|
|
},
|
|
"https://www.google-analytics.com/",
|
|
],
|
|
"frame-ancestors": ["'none'"],
|
|
"style-src": [
|
|
"'unsafe-inline'",
|
|
{
|
|
prod: "https://www.quivr.app/",
|
|
preview: "https://preview.quivr.app/",
|
|
local: ["http://localhost:3000", "http://localhost:3001"],
|
|
},
|
|
],
|
|
};
|
|
|
|
// Resolve environment-specific CSP values
|
|
for (const directive of Object.values(ContentSecurityPolicy)) {
|
|
for (const [index, resource] of directive.entries()) {
|
|
if (typeof resource === "string") {
|
|
continue;
|
|
}
|
|
directive[index] = resource[process.env.NEXT_PUBLIC_ENV];
|
|
if (Array.isArray(directive[index])) {
|
|
directive[index] = directive[index].join(" ");
|
|
}
|
|
}
|
|
}
|
|
|
|
// Build CSP string
|
|
const cspString = Object.entries(ContentSecurityPolicy)
|
|
.map(([key, values]) => `${key} ${values.join(" ")};`)
|
|
.join(" ");
|
|
|
|
// Define headers
|
|
const securityHeaders = [
|
|
{
|
|
key: "Content-Security-Policy",
|
|
value: cspString,
|
|
},
|
|
{
|
|
key: "Referrer-Policy",
|
|
value: "origin-when-cross-origin",
|
|
},
|
|
{
|
|
key: "X-Frame-Options",
|
|
value: "SAMEORIGIN",
|
|
},
|
|
{
|
|
key: "X-Content-Type-Options",
|
|
value: "nosniff",
|
|
},
|
|
{
|
|
key: "X-DNS-Prefetch-Control",
|
|
value: "on",
|
|
},
|
|
{
|
|
key: "Permissions-Policy",
|
|
value: "camera=(), microphone=(), geolocation=(), interest-cohort=()",
|
|
},
|
|
{
|
|
key: "Strict-Transport-Security",
|
|
value: "max-age=31536000",
|
|
},
|
|
];
|
|
//AJouter le content security policy uniquement en pre-vew et en prod
|
|
|
|
// Check if the SENTRY_DSN environment variable is defined
|
|
if (process.env.SENTRY_DSN) {
|
|
// SENTRY_DSN exists, include Sentry configuration
|
|
const { withSentryConfig } = require("@sentry/nextjs");
|
|
|
|
module.exports = withSentryConfig(
|
|
nextConfig,
|
|
{
|
|
// For all available options, see:
|
|
// https://github.com/getsentry/sentry-webpack-plugin#options
|
|
|
|
// Suppresses source map uploading logs during build
|
|
silent: true,
|
|
|
|
org: "quivr-0f",
|
|
project: "javascript-nextjs",
|
|
},
|
|
{
|
|
// For all available options, see:
|
|
// https://docs.sentry.io/platforms/javascript/guides/nextjs/manual-setup/
|
|
|
|
// Upload a larger set of source maps for prettier stack traces (increases build time)
|
|
widenClientFileUpload: true,
|
|
|
|
// Transpiles SDK to be compatible with IE11 (increases bundle size)
|
|
transpileClientSDK: true,
|
|
|
|
// Routes browser requests to Sentry through a Next.js rewrite to circumvent ad-blockers (increases server load)
|
|
tunnelRoute: "/monitoring",
|
|
|
|
// Hides source maps from generated client bundles
|
|
hideSourceMaps: true,
|
|
|
|
// Automatically tree-shake Sentry logger statements to reduce bundle size
|
|
disableLogger: true,
|
|
}
|
|
);
|
|
} else {
|
|
// SENTRY_DSN does not exist, export nextConfig without Sentry
|
|
module.exports = nextConfig;
|
|
}
|