-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathnext.config.js
239 lines (218 loc) · 8.34 KB
/
next.config.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
const withOffline = require('next-offline')
const withImages = require('next-images')
// Next.js might support imports by default soon. Until then, it
// proved annoying to whitelist every module that needs transpiling,
// so we've locked the following pacakges to older versions:
// unified, remark, remark-rehype, rehype-react, rehype-sanitize
// https://github.com/vercel/next.js/issues/9607#issuecomment-944156493
// ---
// Transpile some dependencies.
// https://github.com/vercel/next.js/issues/25454#issuecomment-903513941
// https://github.com/vercel/next.js/issues/9607
// Adding all of `unified`'s dependencies:
// https://github.com/unifiedjs/unified/blob/main/package.json#L48
const withTM = require('next-transpile-modules')([
// 'unified',
//
// // unified's dependencies
// '@types/unist',
// 'bail',
// 'extend',
// 'is-buffer',
// 'is-plain-obj',
// 'trough',
// 'vfile',
])
// Sentry error logging. See:
// https://github.com/vercel/next.js/blob/canary/examples/with-sentry-simple/next.config.js
// Use the hidden-source-map option when you don't want the source maps to be
// publicly available on the servers, only to the error reporting
const withSourceMaps = require('@zeit/next-source-maps')({
devtool: 'hidden-source-map',
})
const basePath = process.env.NEXT_PUBLIC_URLS_BASE_PATH || ''
const generalizedVercelDomain = '.*-gladly-team.vercel.app'
const devAssetsRegex = 'https://prod-tab2017-media.gladly.io/.*'
const prodAssetsRegex = 'https://dev-tab2017-media.gladly.io/.*'
const devCloudFrontRegex = 'https://dev-tab2017.gladly.io/newtab/.*'
const prodCloudFrontRegex = 'https://tab.gladly.io/newtab/.*'
const cachingRegex = new RegExp(
`${generalizedVercelDomain}${basePath}.*|${devAssetsRegex}|${prodAssetsRegex}|${devCloudFrontRegex}|${prodCloudFrontRegex}`
)
// Use the SentryWebpack plugin to upload the source maps during build.
const SentryWebpackPlugin = require('@sentry/webpack-plugin')
const nextConfig = {
webpack5: false,
future: {
webpack5: false,
},
// For routing, we need:
// * the app to live on the /newtab base path.
// * to enforce a trailing slash on pages. This ensures a functional
// service worker and matches the URL in the Tab browser extension.
// * the /v4 path to rewrite to the /newtab path. This allows us to
// access the API from a different base path, which is important
// for routing through CloudFront with the legacy app.
basePath,
trailingSlash: true,
async rewrites() {
return [
{
// This includes the base path: /newtab/service-worker.js
source: '/service-worker.js',
destination: '/_next/static/service-worker.js',
},
// Note: the vercel.json redirect from /v4/(.*) to /newtab/$1 is
// deprecated. Instead, we're accessing the v4 API via
// /newtab/api/, which we added as a CloudFront cache behavior.
]
},
async redirects() {
return [
// Redirect from the index page to the base path index.
// This is for convenience in local development and when
// viewing preview deployments. It shouldn't need to be used
// in production.
basePath && {
source: `/`,
destination: `${basePath}/`,
basePath: false,
permanent: false,
},
{
source: `/uninstalled/`,
destination: 'https://forms.gle/2q33AzpTA3A6vvZo6',
permanent: false,
},
{
source: `/first-tab/`,
destination: `/`,
permanent: false,
},
].filter(Boolean)
},
async headers() {
return [
{
// This includes the base path: /newtab/service-worker.js
source: '/service-worker.js',
headers: [
{
key: 'Cache-Control',
value: 'no-cache',
},
{
key: 'Service-Worker-Allowed',
value: `${basePath}/`, // e.g., /newtab/
},
],
},
]
},
webpack: (config, options) => {
// Sentry error logging. See:
// https://github.com/vercel/next.js/blob/canary/examples/with-sentry-simple/next.config.js
//
// In `pages/_app.js`, Sentry is imported from @sentry/node. While
// @sentry/browser will run in a Node.js environment, @sentry/node will use
// Node.js-only APIs to catch even more unhandled exceptions.
//
// This works well when Next.js is SSRing your page on a server with
// Node.js, but it is not what we want when your client-side bundle is being
// executed by a browser.
//
// Luckily, Next.js will call this webpack function twice, once for the
// server and once for the client. Read more:
// https://nextjs.org/docs#customizing-webpack-config
//
// So ask Webpack to replace @sentry/node imports with @sentry/browser when
// building the browser's bundle
if (!options.isServer) {
// eslint-disable-next-line no-param-reassign
config.resolve.alias['@sentry/node'] = '@sentry/browser'
}
// When all the Sentry configuration env variables are available/configured
// The Sentry webpack plugin gets pushed to the webpack plugins to build
// and upload the source maps to sentry.
// This is an alternative to manually uploading the source maps
// Note: This is disabled in development mode.
if (
process.env.NEXT_PUBLIC_SENTRY_DSN &&
process.env.SENTRY_ORG &&
process.env.SENTRY_PROJECT &&
process.env.SENTRY_AUTH_TOKEN &&
process.env.SENTRY_UPLOAD_SOURCE_MAPS_ON_BUILD === 'true'
) {
config.plugins.push(
new SentryWebpackPlugin({
include: '.next',
ignore: ['node_modules'],
urlPrefix: '~/_next',
release: options.buildId,
})
)
}
return config
},
// Modify our service worker manifest.
transformManifest: (manifest) => ['/'].concat(manifest), // add the homepage to the cache
// Let us manually register the service worker.
// https://github.com/hanford/next-offline#runtime-registration
dontAutoRegisterSw: true,
// Whether to enable the service worker in development. Note this may not work
// if we don't have a custom _error.js file:
// https://github.com/hanford/next-offline/issues/190#issuecomment-535278921
generateInDevMode: false,
// The base path from which to serve the service worker. This affects the
// default scope.
// https://developers.google.com/web/ilt/pwa/introduction-to-service-worker#registration_and_scope
registerSwPrefix: '/newtab',
// Limit the service worker to this app's base path.
scope: '/newtab/',
workboxOpts: {
swDest: 'static/service-worker.js',
cleanupOutdatedCaches: true,
clientsClaim: true,
skipWaiting: true,
// Cache strategies for different resources:
// https://developers.google.com/web/tools/workbox/modules/workbox-strategies#using_strategies
runtimeCaching: [
{
// All resources except requests to /api/* or /graphql*, including
// variants with our base path. Note that our base path, "/newtab", is
// hardcoded here.
// https://regex101.com/r/5cs6L7/1/tests
// see caching strategies here https://developers.google.com/web/tools/workbox/modules/workbox-strategies
urlPattern: cachingRegex,
handler: 'StaleWhileRevalidate',
options: {
cacheName: 'tab-resources',
expiration: {
maxEntries: 500,
maxAgeSeconds: 30 * 24 * 60 * 60, // 1 month
// Automatically clean up if quota is exceeded.
purgeOnQuotaError: true,
},
cacheableResponse: {
statuses: [0, 200],
},
},
},
{
// Requests to /api/* or /graphql*, including variants with our base
// path. Note that our base path, "/newtab", is hardcoded here. With a
// "network only" strategy, this should be the same as not defining
// any caching at all, so we're just being explicit here.
// https://regex101.com/r/2ttcQE/2
urlPattern:
/^http[s]?:\/\/(?:[^/\s]+\/)(?:(api\/|graphql(?:\/)?$|newtab\/api\/|newtab\/graphql(?:\/)?$))/,
handler: 'NetworkOnly',
},
],
},
// Automatically inline all images under this size.
inlineImageLimit: 16384,
}
// withTM should be outermost plugin.
// https://github.com/martpie/next-transpile-modules
module.exports = withTM(withSourceMaps(withOffline(withImages(nextConfig))))