Skip to content

Commit 655def5

Browse files
Move serviceworker to workbox and fix SSE interference (#11538) (#11547)
* Move serviceworker to workbox and fix SSE interference Instead of statically hardcoding every frontend asset, this uses a type-based approach to cache all js,css and manifest.json requests. This also fixes the issue that the service worker was interfering with EventSource because it was unconditionally handling all requests which this new implementation doesn't. Fixes: #11092 Fixes: #7372 * rethrow error instead of logging * await .register * Revert "rethrow error instead of logging" This reverts commit 043162b. * improve comment * remove JSRenderer * add version-based cache invalidation * refactor * more refactor * remove comment * rename item to fit cache name Co-authored-by: guillep2k <[email protected]>
1 parent 2042cf2 commit 655def5

File tree

12 files changed

+96
-133
lines changed

12 files changed

+96
-133
lines changed

.eslintrc

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,7 @@ rules:
4848
no-cond-assign: [2, except-parens]
4949
no-console: [1, {allow: [info, warn, error]}]
5050
no-continue: [0]
51+
no-empty: [2, {allowEmptyCatch: true}]
5152
no-eq-null: [2]
5253
no-mixed-operators: [0]
5354
no-multi-assign: [0]

modules/templates/dynamic.go

Lines changed: 0 additions & 12 deletions
Original file line numberDiff line numberDiff line change
@@ -48,18 +48,6 @@ func JSONRenderer() macaron.Handler {
4848
})
4949
}
5050

51-
// JSRenderer implements the macaron handler for serving JS templates.
52-
func JSRenderer() macaron.Handler {
53-
return macaron.Renderer(macaron.RenderOptions{
54-
Funcs: NewFuncMap(),
55-
Directory: path.Join(setting.StaticRootPath, "templates"),
56-
AppendDirectories: []string{
57-
path.Join(setting.CustomPath, "templates"),
58-
},
59-
HTMLContentType: "application/javascript",
60-
})
61-
}
62-
6351
// Mailer provides the templates required for sending notification mails.
6452
func Mailer() (*texttmpl.Template, *template.Template) {
6553
for _, funcs := range NewTextFuncMap() {

modules/templates/static.go

Lines changed: 0 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -132,15 +132,6 @@ func JSONRenderer() macaron.Handler {
132132
})
133133
}
134134

135-
// JSRenderer implements the macaron handler for serving JS templates.
136-
func JSRenderer() macaron.Handler {
137-
return macaron.Renderer(macaron.RenderOptions{
138-
Funcs: NewFuncMap(),
139-
TemplateFileSystem: NewTemplateFileSystem(),
140-
HTMLContentType: "application/javascript",
141-
})
142-
}
143-
144135
// Mailer provides the templates required for sending notification mails.
145136
func Mailer() (*texttmpl.Template, *template.Template) {
146137
for _, funcs := range NewTextFuncMap() {

package-lock.json

Lines changed: 22 additions & 0 deletions
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -47,6 +47,8 @@
4747
"webpack": "4.43.0",
4848
"webpack-cli": "3.3.11",
4949
"webpack-fix-style-only-entries": "0.4.0",
50+
"workbox-routing": "5.1.3",
51+
"workbox-strategies": "5.1.3",
5052
"worker-loader": "2.0.0"
5153
},
5254
"devDependencies": {

routers/routes/routes.go

Lines changed: 0 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -1048,10 +1048,6 @@ func RegisterRoutes(m *macaron.Macaron) {
10481048
ctx.HTML(200, "pwa/manifest_json")
10491049
})
10501050

1051-
m.Get("/serviceworker.js", templates.JSRenderer(), func(ctx *context.Context) {
1052-
ctx.HTML(200, "pwa/serviceworker_js")
1053-
})
1054-
10551051
// prometheus metrics endpoint
10561052
if setting.Metrics.Enabled {
10571053
c := metrics.NewCollector()

templates/base/head.tmpl

Lines changed: 2 additions & 24 deletions
Original file line numberDiff line numberDiff line change
@@ -6,30 +6,6 @@
66
<meta http-equiv="x-ua-compatible" content="ie=edge">
77
<title>{{if .Title}}{{.Title | RenderEmojiPlain}} - {{end}} {{if .Repository.Name}}{{.Repository.Name}} - {{end}}{{AppName}} </title>
88
<link rel="manifest" href="{{AppSubUrl}}/manifest.json" crossorigin="use-credentials">
9-
{{if UseServiceWorker}}
10-
<script>
11-
if ('serviceWorker' in navigator) {
12-
navigator.serviceWorker.register('{{AppSubUrl}}/serviceworker.js').then(function(registration) {
13-
// Registration was successful
14-
console.info('ServiceWorker registration successful with scope: ', registration.scope);
15-
}, function(err) {
16-
// registration failed :(
17-
console.info('ServiceWorker registration failed: ', err);
18-
});
19-
}
20-
</script>
21-
{{else}}
22-
<script>
23-
if ('serviceWorker' in navigator) {
24-
navigator.serviceWorker.getRegistrations().then(function(registrations) {
25-
registrations.forEach(function(registration) {
26-
registration.unregister();
27-
console.info('ServiceWorker unregistered');
28-
});
29-
});
30-
}
31-
</script>
32-
{{end}}
339
<meta name="theme-color" content="{{ThemeColorMetaTag}}">
3410
<meta name="author" content="{{if .Repository}}{{.Owner.Name}}{{else}}{{MetaAuthor}}{{end}}" />
3511
<meta name="description" content="{{if .Repository}}{{.Repository.Name}}{{if .Repository.Description}} - {{.Repository.Description}}{{end}}{{else}}{{MetaDescription}}{{end}}" />
@@ -84,8 +60,10 @@
8460
</script>
8561
<script>
8662
window.config = {
63+
AppVer: '{{AppVer}}',
8764
AppSubUrl: '{{AppSubUrl}}',
8865
StaticUrlPrefix: '{{StaticUrlPrefix}}',
66+
UseServiceWorker: {{UseServiceWorker}},
8967
csrf: '{{.CsrfToken}}',
9068
HighlightJS: {{if .RequireHighlightJS}}true{{else}}false{{end}},
9169
Minicolors: {{if .RequireMinicolors}}true{{else}}false{{end}},

templates/pwa/serviceworker_js.tmpl

Lines changed: 0 additions & 83 deletions
This file was deleted.

web_src/js/features/serviceworker.js

Lines changed: 43 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
const {UseServiceWorker, AppSubUrl, AppVer} = window.config;
2+
const cacheName = 'static-cache-v2';
3+
4+
async function unregister() {
5+
for (const registration of await navigator.serviceWorker.getRegistrations()) {
6+
const serviceWorker = registration.active;
7+
if (!serviceWorker) continue;
8+
registration.unregister();
9+
}
10+
}
11+
12+
async function invalidateCache() {
13+
await caches.delete(cacheName);
14+
}
15+
16+
async function checkCacheValidity() {
17+
const cacheKey = AppVer;
18+
const storedCacheKey = localStorage.getItem('staticCacheKey');
19+
20+
// invalidate cache if it belongs to a different gitea version
21+
if (cacheKey && storedCacheKey !== cacheKey) {
22+
invalidateCache();
23+
localStorage.setItem('staticCacheKey', cacheKey);
24+
}
25+
}
26+
27+
export default async function initServiceWorker() {
28+
if (!('serviceWorker' in navigator)) return;
29+
30+
if (UseServiceWorker) {
31+
await checkCacheValidity();
32+
try {
33+
await navigator.serviceWorker.register(`${AppSubUrl}/serviceworker.js`);
34+
} catch (err) {
35+
console.error(err);
36+
await invalidateCache();
37+
await unregister();
38+
}
39+
} else {
40+
await invalidateCache();
41+
await unregister();
42+
}
43+
}

web_src/js/index.js

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ import initGitGraph from './features/gitgraph.js';
1515
import initClipboard from './features/clipboard.js';
1616
import initUserHeatmap from './features/userheatmap.js';
1717
import initDateTimePicker from './features/datetimepicker.js';
18+
import initServiceWorker from './features/serviceworker.js';
1819
import {initTribute, issuesTribute, emojiTribute} from './features/tribute.js';
1920
import createDropzone from './features/dropzone.js';
2021
import highlight from './features/highlight.js';
@@ -2477,6 +2478,7 @@ $(document).ready(async () => {
24772478
initGitGraph(),
24782479
initClipboard(),
24792480
initUserHeatmap(),
2481+
initServiceWorker(),
24802482
]);
24812483
});
24822484

web_src/js/serviceworker.js

Lines changed: 16 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
import {registerRoute} from 'workbox-routing';
2+
import {StaleWhileRevalidate} from 'workbox-strategies';
3+
4+
const cacheName = 'static-cache-v2';
5+
6+
const cachedDestinations = new Set([
7+
'manifest',
8+
'script',
9+
'style',
10+
'worker',
11+
]);
12+
13+
registerRoute(
14+
({request}) => cachedDestinations.has(request.destination),
15+
new StaleWhileRevalidate({cacheName}),
16+
);

webpack.config.js

Lines changed: 8 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -35,13 +35,20 @@ module.exports = {
3535
jquery: [
3636
resolve(__dirname, 'web_src/js/jquery.js'),
3737
],
38+
serviceworker: [
39+
resolve(__dirname, 'web_src/js/serviceworker.js'),
40+
],
3841
icons: glob('node_modules/@primer/octicons/build/svg/**/*.svg'),
3942
...themes,
4043
},
4144
devtool: false,
4245
output: {
4346
path: resolve(__dirname, 'public'),
44-
filename: 'js/[name].js',
47+
filename: ({chunk}) => {
48+
// serviceworker can only manage assets below it's script's directory so
49+
// we have to put it in / instead of /js/
50+
return chunk.id === 'serviceworker' ? '[name].js' : 'js/[name].js';
51+
},
4552
chunkFilename: 'js/[name].js',
4653
},
4754
optimization: {

0 commit comments

Comments
 (0)