I use create create app for teaching and I don't see the benefit in making stuff more complicated with adding so much advanced boilerplate code.
You probably know that service worker won't work on any iOS device (20% - 25% of all your users) and this is not about to change soon (maybe never).
Adding service workers by default is about to create very different experiences and bug behaviour depending on the device. It also makes it way more complicated to get a create react app to work offline for all platforms.
Please don't turn cra into something like a Windows98 computer where you had to remove AOL and other preinstalled stuff before being able to be productive.
cc @jeffposnick
I see your point but I don't agree removing a single line from src/index.js
is a big deal (and that's all you need to do to opt out). If you can clarify what exactly makes your life harder (in practical terms, e.g. a specific scenario that is confusing) maybe we can make it better via other means.
While iOS is lagging behind, there is more momentum than ever behind using service workers for webapps (basically everywhere except iOS), and weâre going to stick with this for a while. If people donât find it useful and most of the feedback we get repeats yours, we will remove it. So far the reception has been positive, but itâs hard to say until we live with this change for at least several months.
If the tool was called âcreate react websiteâ then Iâd agree with you, but people expect more responsiveness from apps, and precaching helps bridge that gap. Weâll see if it works out or not.
This is an excellent response @gaearon. I was initially quite surprised to see it there (I found this thread searching for effectively the exact title of the issue!), but my original reaction was largely because _I still don't feel super comfortable using SWs_. I know plenty about them, but I don't feel like I _really_ understand all the potential trapsâI don't know what I don't know, you know? ;)
But the momentum behind PWAs is undeniable, and if people's first concrete introduction to SWs is that their CRA-ed apps are precached (and if they don't want it, disabling it is a one-liner) then this sounds like a super smart addition đ
We've used PWA and service workers for museum mobile webapps for a year and it has been great. I think encouraging their use is absolutely what we should be doing.
Here is example of a person wasting hours (and even deleting Node) being confused by the cache. I'm still on board with service workers but there is a clear usability/discoverability issue here.
@jeffposnick Do you have thoughts on how to make troubleshooting these scenarios less maddening? We canât keep this as a default if people are going to have experiences like this.
https://www.reddit.com/r/javascript/comments/6dwbig/react_running_even_after_everything_removed/
One possible solution is to show "The app was served from [offline cache]" right in the default App.js when it happens. This would make the feature more discoverable. Can we detect this?
Why not just make it an option (default off)? I doubt many newbies are going to absolutely need service workers and experience devs who want to use CRA to start developing new projects quickly will know how to have it easily create their app with SW support. Seems like a win:win to me.
One possible solution is to show "The app was served from [offline cache]" right in the default App.js when it happens. This would make the feature more discoverable. Can we detect this?
@gaearon it's possible, I came up with a solution a while ago: https://github.com/lukechilds/onionite/commit/416ccb07e7a006e679ddc86f93bcb30cdd7cac1b#diff-7a7a9fab4ca3224cad19254b6aba0be3R60
It's pretty ugly though, I'm sure there's a cleaner way.
You can demo that commit by viewing https://onionite.now.sh, then disable network and view a SW cached page. It should say something like *There seems to be an issue connecting to the server. This data is cached from Fri Jan 13 2017 15:00:38 GMT+0700 (ICT)
at the top of the page.
Service workers are intentionally super sticky, so even if you remove the one line if you ran the app before it's now there, and you need to write code to remove it from you browser (or know where the knobs are in the chrome dev tools).
Either way tho you are immediately asking folk to learn/understand an advanced low level api before even starting to build an app.
I was also confused by that file and I'm not a total newbie. Maybe it would be helpful to include a description on how to opt out, instead of directing users to a PWA section of the readme?
The only thing I'm asking in that thread is that we try to focus on the solutions instead of dismissing the idea outright. If the problem is "Chrome toggles are tucked away too far", Chrome folks are invested in solving this and we can influence that. If the problem is lack of a comment we can add it. Et cetera. Let's just entertain the thought that it might be possible to make this a good default, and then see what are the paper cuts, and how to go about them.
@shime I think the README is a perfect place for this information. There are no comments for importing CSS files inside a .js
file and that itself is a whole new concept to some people.
Removing the Service Worker file isn't too much hard work though. @gaearon is right - we should see how PWA's progress and it's great there's built in support for this functionality.
Sorry I'm not trying to be dismissive :) I don't think tho this is really just a UX issue. Having the SW register script causes odd issues for folks that aren't familiar with it while also offering minimal value in the PWA arena out of the box, because the content of the SW is deeply app specific and the caching rules ultra dependent on the dev understanding the cache pattern that suits each particular request, resource etc. Otherwise you get frustration and issues.
That said from a DX perspective I've almost always want the SW off in development unless I'm specifically working on offline flows or fallback. In which case tho it's important to run in DEV mode otherwise it's too much time between builds to tweak and troubleshoot. This ends up being an issue mostly with chrome burying the SW toggles in the application tab
The other issue here is that just preaching assets is troublesome, some assets I don't want to cache while others I want a ttl or something. Webpack makes this harder because with hashes you don't know the name of assets so you can even adjust or tweak the rules there (does precache plugin even allow that). Generally I've had good success with webpack-service-worker-plugin as it uses the manifest to make assets available in the service worker, and runs it through webpack so you can still use Babel, etc in the file
The SW behaviour takes time to understand until you super comfortable with SW behaviour, work in an incognito browser window, that will help your browser forget about all SW on that localhost port and start fresh always.
SW API is powerful and we can programmatically do a lot of things, opt-out and clean everything is doable really easy. Knowing what's cached is easy, we just need to listen and learn from Devs what issues are they facing so we ship the right and useful SW Dev util.
Example of a nice experiment I like to clean SW and local cache and has useful application on dev:
https://gist.github.com/gauntface/38fcbd5495455c06f45d2b674f57be83
SW is more lovely than what it seem to be, bit of education about it and right helpers will keep experience great for devs and users!
If there's a standard way of displaying overlays/toasts in c-r-a, then methods of the Service Worker API can be used to notify developers what's going on.
There's a very basic placeholder in the registration script that logs something to the JS Console, but that's (intentionally) not in-your-face.
Code like
if ('serviceWorker' in navigator &&
process.env.NODE_ENV !== 'production' &&
navigator.serviceWorker.controller) {
// Display some sort of message about the page being loaded by the service worker.
}
could be used to further inform developers while outside of the production
environment.
The sw-precache-webpack-plugin
does support a blacklist of assets that shouldn't be precached, but that involves modifying the Webpack configuration, which in turn implies an eject
.
The majority of the PR to add in PWA support consisted of explanatory text in the README
in an attempt to explain various implications of using a cache-first strategy, but this is obviously a big change in the development and deployment model. Getting information in front of developers via overlays and contextual messages is definitely a good idea if there's a standard way of doing it.
Either way tho you are immediately asking folk to learn/understand an advanced low level api before even starting to build an app.
This is valuable feedback. Letâs address it. There are a few different places we can add proactive developer education that can help.
Toasts - a visible, teachable moment
Across a few different boilerplates, weâve found toasts to be a low-friction way to communicate to both users and developers that a page is offline capable. Here are some examples:
If you also try out Voice Memos or SVGOMG you'll see that these toasts have a finite lifetime and are only shown for a few seconds.
For a project like create-react-app, such a toast could even link out the text in the toast to the CRA Service Worker documentation and we could use that as another teaching moment to help beginners learn what Service Workers are giving them.
I'll leave it to the more savvy designers on CRA to suggest ideas, but here's just one take on it:
This is also helpful for a few reasons:
Sometimes folks feel a toast requires bringing in a whole UI library. It can be achieved in just a few lines of code independently and if there's interest we can work on a PR for what this would look like.
Runtime Error Message overlay/Better console messages
Inform developers that the page is offline capable and provide some short text on where to learn more about CRA's support for it and where they can learn more about debugging Service Workers (this codelab has been great) . I'm unsure if the runtime overlay is strictly only meant to be used for DEV development messages, but that could be another option.
We can be strategic about linking out to answers for the most common questions here regardless of whether it's shown in an overlay or the console.
For SW specifically, anything we can do to teach devs about the existence of the Application panel is good. A lot of the time you'll just use it to verify: 1) your offline support is indeed working as expected, 2) you want to be extra sure the freshest content is being served, so 'Clear Storage' is your friend.
React Developer Tools
We could define an additional check as part of the current production messaging (or somewhere inside the React DevTools panel itself) a way to inform developers that their page is offline capable and was served using Service Workers.
An additional check/entry here could be "This page appears to be offline capable. Learn more about this" that links up to the CRA SW documentation.
Chrome DevTools
This is a longer-term change, but I want to lean on the team on our side to provide very visible indicators that a Service Worker is active on the page. I'm unsure if we accomplish this using a butter-bar, or our own overlays, but it's a common enough request that I think we should do something better to inform beginners that offline support has kicked in. I'll talk to the team and update this thread with any ideas we brainstorm.
This is a longer-term change, but I want to lean on the team on our side to provide very visible indicators that a Service Worker is active on the page. I'm unsure if we accomplish this using a butter-bar, or our own overlays, but it's a common enough request that I think we should do something better to inform beginners that offline support has kicked in.
This sounds like the best way to go about it, and will benefit any setups (not just CRA).
If you want PWA approach to gain traction, it is important to teach developers to trust this feature. Any time I use a tool that caches something too aggressively, a single mistake or misunderstanding is enough to forever destroy my trust and learn to either wipe away or disable the cache. Itâs not very rational, but itâs how human mind works: once burned, youâll avoid the feature as much as you can.
Now that service workers are going mainstream, it is important not to make peopleâs first experiences bad. Otherwise the trust will never be recovered, and even if the idea is good, the execution will botch their adoption.
I think it's very important that @gaearon didn't dismiss the problem on reddit and turned this into an opportunity. SW are great, but there was a whole talk about how dangerous it can be at JSConf. (https://www.youtube.com/watch?v=CPP9ew4Co0M)
Possibly as a first step, SW may not work at all in development mode. Offline indicators should also be a part of the app, but design is challenging.
Just to make it clear, SWs are not enabled in development mode in CRA.
My bad, after reading the reddit post I assumed it was. Will be more careful next time. :)
We could turn service workers off by default for localhost?
As the only real reason to have them on for localhost is if you were actually testing SW and therefore you would hopefully be able to find the how to turn them on for localhost again.
Thanks again for create-react-app, I use it all the time.
The solution I think is to carefully document new features that are added to CRA. Something like these points in the doc would have been very helpful.
SERVICE WORKERS have been added to CRA: (required reading)
Edit: just to be clear, I think it is great service workers were added! I just wish I had been warned that I didn't know what I didn't know.
I'm excited about service workers and glad to see CRA integrating them. But I'm not sure the current scaffolding is rich enough to turn on by default. Why not just have the SW commented out by default in index.js and making it opt in until the legit issues have been better addressed and more education can be done?
Here's one problem my app ran into when I added SWs a couple months ago. I have a real-time app which really requires everyone using it together at the same time to be on the same version. Historically new releases were never a problem. When I added SWs because of the need for every return user to hard reset to update the cache, some people were on, say, 4.0 and some on 5.0. This broke both user expectations and the app itself.
This was easy to catch in QA, but some problems are (1) it's not something that should be worried about in, say, an early version of an app, (2) it's one more thing, and a low level thing at that, for beginners to account for, (3) a good solution is not obvious, (4) the solution is probably generic across apps and thus should be at least documented in CRA and probably with some overridable toast handling.
Again, I'm grateful for the work everyone is doing integrating SWs into CRA, just a case to think about.
I just ran into another SW issue.
I was using serve
to run my build locally on localhost:5000
, then I used serve
again to host a different site on localhost:5000
. But because of the service worker, i received the old site instead of the new one.
If I didn't already know about CRA, SW and this thread, I would have teared my hair out trying to figure that one out.
Which takes me back to the idea of not having service workers running for anything on localhost
, or even turning off by default might be a good option for now.
I was using serve to run my build locally on localhost:5000, then I used serve again to host a different site on localhost:5000. But because of the service worker, i received the old site instead of the new one.
I also got a similar issue a few times and found it annoying and confusing.
@jeffposnick @addyosmani
Do you see any specific solution to this, from developer experience point of view? Itâs not even about toasts and indicating an app is offline and cachedâitâs about different apps usually running on the same port locally one after another.
Which takes me back to the idea of not having service workers running for anything on localhost
Iâm worried this will lead to extreme confusion when the app behaves differently in production.
I thought we already handled that case once you refresh once (with our noop service worker); I'm not sure if it gets much better than that.
Thatâs for running different production apps.
For example if I do npm run build
for different React apps and then attempt to serve
them.
There are a few different scenarios related to localhost
development/staging:
npm start
for development, and serve
for staginglocalhost:3000
should not end up with any SW registered, and development will continue as before. localhost:5000
will end up with the SW registered, and the cache-first strategy means that if you switch from one project to another, you'll see the old project the next time you visit localhost:5000
âmeanwhile the new project will be cached in the background, and then the next time you visit after that, you'll see the new project's output. (I'm assuming that if @ro-savage had known to reload localhost:5000
, they would have seen the content they expectedâif not, that's a bug I would definitely like to follow up on.)
Currently, when this happens, there should be a message logged in the JS console explaining that new content is available, but this only visible if you know where to look.
An overlay or some sort of other notification can help here in terms of letting developers know that there's new content that was downloaded in the background. This might be on by default for localhost
and perhaps optional for a real server (in case the developer wants to show their own UI elements instead.)
I'm really not one for crafting React UX elements, but I think all the right hooks are there to trigger the displayâit's just a question of having a standard way of showing it (other than console.log()
).
I agree that disabling SW entirely on localhost
would not be ideal, because being able to confirm the SW behavior prior to deployment is important.
localhost:3000
, and then you use npm start
This scenario should be addressed by this PR.
@jeffposnick - The console.log()
message only seems to come up if the page is another create-react-app.
If you are running a non CRA site/app on the same localhost:port
not only won't you get the message but refreshing doesn't load the new page. You actually have to de-register the service worker manually to see the correct content.
Is there a way within the client app itself we can give the service worker a permanent ID and everytime it starts, it checks to see if 1) there is a service worker 2) it has the same ID?
If it gets no response, check if it gets a 200 ok
from the server. On 200 ok, force refresh. On timeout do nothing.
If it gets a response with a different ID, force refresh.
I wasn't able to find an id field for service workers (at least not one that worked).
But if we gave the service-worker.js a unique ID generated when the project was first created we could do something like this.
export default function register() {
if (process.env.NODE_ENV === 'production' && 'serviceWorker' in navigator) {
window.addEventListener('load', () => {
// this would become service-work-id.js
const swUrl = `${process.env.PUBLIC_URL}/service-worker.js`
if (!navigator.serviceWorker.controller) {
// No service worker yet
registerServiceWorker(swUrl)
} else {
fetch(swUrl).then(res => {
// Check to see if the SW URL is valid
if (res.ok) {
// Matches. All good. Continue with registering SW
registerServiceWorker(swUrl)
} else {
// SW URL was invalid.
fetch(`${window.location.protocol}//${window.location.host}`).then(res2 => {
// Just check if online
if (res2.ok) {
// Unregister and refresh page
unregister()
window.location.reload(true)
} else {
console.log('Offline. Using cached copy')
}
})
}
})
}
})
}
}
function registerServiceWorker(url) {
navigator.serviceWorker
.register(url)
.then(registration => {
console.log('reg.scope', registration.scope)
registration.onupdatefound = () => {
const installingWorker = registration.installing
installingWorker.onstatechange = () => {
if (installingWorker.state === 'installed') {
if (navigator.serviceWorker.controller) {
// At this point, the old content will have been purged and
// the fresh content will have been added to the cache.
// It's the perfect time to display a "New content is
// available; please refresh." message in your web app.
console.log('New content is available; please refresh.')
} else {
// At this point, everything has been precached.
// It's the perfect time to display a
// "Content is cached for offline use." message.
console.log('Content is cached for offline use.')
}
}
}
}
})
.catch(error => {
console.log('No service worker found')
console.error('Error during service worker registration:', error)
})
}
export function unregister() {
if ('serviceWorker' in navigator) {
navigator.serviceWorker.ready.then(registration => {
registration.unregister()
})
}
}
@ro-savage The idea of forcing a reload using a deterministic hint is an interesting idea. Have you tested out the pattern above to verify it addresses the non-CRA site being served on the same port as a cached CRA site issue?
An addition I was going to add to @jeffposnick's last list was we could explore port randomization for serving up the production build. I've used this in some of my own local boilerplates.
This would be a naive way of limiting the chances of another site being served on the same port as a previously cached CRA app. However, this may be less desirable than some of the other alternatives we could explore.
@addyosmani - Tested and working locally on mine both with multiple CRAs and with non-CRA sites all on the same port.
I'll put in a PR for people to play with.
I am not entirely sure about what edge cases might exist.
Rather than port randomisation, could we generate hashed local addresses like http://my-app-name-hash1a2b3c4d5e.dev a bit like how Pow works?
Alternatively, http://my-app-name-hash1a2b3c4d5e.localhost:5000 could work too, and wouldn't even require any changes to serve
, but it's actually just being keyed off the port so if you accidentally load up http://localhost:5000 you'd hit the same problem.
Of course, this is all assuming there's no added weirdness with SWs and subdomains...
Oo that's neat, tho I think the problem with that approach might be that SW require https unless it's on localhost
Ah yes of course. Potentially then serve --port 5000 --subdomain my-app-name-hash1a2b3c4d5e
could return a 301 to redirect http://localhost:5000 to the subdomain, so the SW is never installed except for a URL tied to that app & revision.
Added #2438 which is an attempted to invalidate SW cache on both different CRA apps and non-CRA apps when hosted on the same domain+port.
While we could tell the user to use serve
and give them the full command to use. It feels like a work-around rather than a fix. It doesn't solve the issue if they use something else to host locally, or if they try to serve it later and don't remain the correct --subdomain my-app-name-hash1a2b3c4d5e
, etc. Same goes for random ports.
CRA 'should just work'.
@gaearon - What are you feeling about disabling PWA until the issues are solved?
Currently CRA is potentially breaking other apps and projects locally for people and knowing both that its caused by SW and how to unregistered a service worker is quiet confusing.
Two cents: with a solution like #2438 providing a workaround, why don't we try landing something we acknowledge decreases the chances of someone tripping up on SW support without disabling it? We could keep iterating on a better solution here in the mean time.
Continuous feedback on the DX gaps from SW being a default for production forces us to address any pain points sooner rather than later.
Appears to only be affecting Firebase. Will investigate. now
and surge
both seem to refresh fine.
Deploying changes with serve
, now
, surge
and firebase
. Firebase failing to refresh the page with SW, rest are working fine.
Approximately 3 minutes
https://www.youtube.com/watch?v=oYy-WqkgS1Q
Full details see: #2440
I am getting returned an old index.html and bundles from the service workers.
Going to site in chrome, which has been visited previously before latest deploy
Same site in firefox, which hasn't visited previously
Notice the different bundle names. (DevTools open which disabled cache checked + index.html is served with cache-control 0).
On top of that, I can't unregister users because when I update the bundle with the unregister()
instead of registerServiceWorker()
the users never receive the updated bundle. Meaning the only way it for the user to manually delete the service worker?
Thanks to all to lead this question in an open and constructive discussion.
I understand that google uses a lot of dollars and pressure to push service worker into the web so Apple is forced to adapt.
And please don't get me wrong ServiceWorker is an awesome technology and hopefully Apple will abandon their plan of becoming the company which builds the most hated mobile browser.
But I believe it's not about the technology.
For me create-react-app was the perfect start point which had one main goal:
Get all the road blocks out of the way and get excited about react!
Even a less experienced developers gets his/her first react app running without frustration of all the magic which happens behind the scenes.
For me ServiceWorker is just a technology like TypeScript/Flow or Sass.
On the one hand they all add a lot of advantages but on the other hand they add complexity which comes with a lot of possible frustration (as you read in the comments above).
Please keep it simple and don't fight googles fight for service worker here in this project.
As I needed to show UI elements when offline ready, in offline mode etc. so I converted the registerServiceWorker.js into <Serviceworker/>
component, and rendered in App.js.
Showing in browser UI instead of console does makes it very clear. Default "Offline Ready: message can also have unregister button and other messages with link explaining how to change. It can be very much be in your face.
Maybe we can have a default message in development mode also.
Should I make a PR with UI messages?
edit - basically it what @addyosmani suggested in his toasts comment. Looks really clear - no magic - render component to get default functionality with some UI to remind you
@bunshar - @viankakrisna has created PR #2426 for creating a toast.
Maybe take a look at his PR and do a review/leave some comments form your findings. I believe he did it a slightly different way.
@jantimon Thanks for sharing your thoughts, this is appreciated. I think thereâs a middle ground here but we need to explore it a bit further to find it.
I use "production" builds during development - much of my ui depends on a complex back end. There have been some references to triggering service workers based on "production" or "development" and I would like to make sure the solution for SW's allows them to be disabled for production.
@talkingtab you may completely disable service workers by removing the "registerServiceWorker" line in index.js.
Thoughts on PWA / Service Workers
TLDR: Agree with @jantimon. Too many issues. Not enough advantages for majority of users. CRA should 'just work'. Make PWA/SW optional.
Why was CRA created?
(Please correct me if I am wrong @gaearon)
CRA was developed due to the outcry of people who struggled to get started with React. There was boilerplate, webpack setup, babel transformations, and totally new paradigm in React.
It was provide an easy-to-get started application with sensible defaults that just work.
Problems with Service Workers
I consider myself a pretty experienced developer, and have created production ready offline-first webapps using AppCache/Manifest.
I have run into numerous problems since CRA introduced PWA by default. From localhost showing the wrong pages, to production-ready deploys breaking due to cache configuration, to issues with how chrome handles no-cache
options.
And of course, others have encountered issues as can be seen from this thread. People who are confident enough developers to post on the github issues of the project.
Beyond that, Service Workers are new and still experimental. The spec is changing and browsers are playing catch up, with different behaviours. (Obviously this alone isnt enough. We need make a choice, thus why we use babel but also don't use stage-2 and stage-3)
Why were service workers added?
Offline capability and faster loading especially for mobile browsers seems to be a big win. And it totally is for some apps, it can be a huge benefit.
For most apps, its an 'added bonus', especially if comes free and doesn't impact on what people are doing.
Now the faster loading, that's hard to judge, because 'it depends'. With my own projects, I have cache-control setup the way we need and the the difference in negligible. (Bar the difference that PWA is forever-cached then updated, while cache-control requires a full reload after it expires). For projects with no-cache or very short caches, and large file sizes, the are probably substantial speed benefits.
Offline-first is great and now that older AppCache method is being removed from browsers this may be the only way to do it. However, it does require you to think about your app different. ]
Disadvantage of service workers
Many developer won't see much positive difference, but can encounter unexpected and unknown issues.
Some examples:
location:port
as a CRA, currently break and will never be shown without manually deleting the service worker (e.g. localhost:5000 for CRA, and the for a static page). (PR #2438 tries to address this)no-cache
on service workers. (This should get fixed when browsers implement the latest spec. Current only Firefox 53+)Conclusion
None of these disadvantages are so great as to say 'No one should use PWA'. In fact, people should be using it, either for experiment, because you know what you are doing, or because its a big benefit for the type of app you are making.
But should it come as standard and default in create-react-app, whose point is to make it easy to develop in React without having to worry about all these setup issues?
CRA has been taken over a bit by developer like myself, who got tired of maintaining our own setup and thought using CRA would be easier, faster, give a better dev experience, give free updates (yay webpack2!) and let us focus on creating our apps.
But this means suddenly a lot of influence comes from experience developers.
I help teach beginner JS developers at a Bootcamp and meetups. I recommend they use CRA because it's suppose to 'just work'. I feel that Service Workers has partial broken the idea that it just works
.
I don't believe that means it should be removed completely but maybe make it opt in by using the current config and an letting the user add registerServiceWorker().
/my 2cents. Personally, I am going to remove PWA 'on by default' from the react-scripts
fork I maintain for our teams projects + my personal projects.
Wondering about the status on this?
CRA is a "no configuration"(NC) tool. CRA has earned a huge following because users trust that it is exactly that. This comment: 'you may completely disable service workers by removing the "registerServiceWorker" line in index.js' is not part of a "no configuration" system.
There are several options to fix this, but, as several others have said, I would prefer to keep services workers. A way to solve this and future similar issues is to add to the a "npm build experimental" command. I think this fits the CRA paradigm.
Weâll leave this open for a while to let more people express their feedback.
I agree that Googleâs own products having issues (Chrome, Firebase) is an indicator that this is not ready to be the default. I want to hear from more people about this, and especially for someone to compile a list of problems that need to be fixed for it to become a good default.
I think itâs likely weâll change from default to easy opt-in in future versions because of feedback in this thread. But if we do, we need a clearly described list of issues which, when fixed (whether by Chrome, Firebase, etc), are enough to turn it back on.
Please keep the constructive feedback coming!
Other frameworks, like Ember, have solved this very problem allowing the development of addons. For instance, ember init
creates a core project that works out-of-the-box while allowing the users to add more capabilities (like validation, offline support, authentication) by installing addons.
I think this project will benefit from that experience.
IMO, Service Workers should be an option and not part of the default scaffolded project.
@eabait it I don't think that makes sense for CRA. It makes sense for Ember which is a more curated framework from the get-go, while React is a much more community-grown affair and needs something that provides a sensible default.
React/webpack etc is very pluggable, you just have to eject. Creating a meta-plugin system that can operate pre-eject seems like a huge can of worms to me, and also misunderstanding the main offering of CRA.
CRA is not a scaffolding system like Yeoman. What CRA provides is a firm, predictable default that offers a shared reality. "I'm using create-react-app version 1.2.3" provides a lot of information that allows others to help you and also makes any kind of tutorial more predictable. If you start adding a plugin structure to it, you start adding different versions of reality and you're torpedoing the biggest selling point of CRA.
Well put @mpj, this is pretty much how I think about it.
@mpj thanks for the clarification. Then, I think SW should not be part of CRA.
I'm all in for service workers. I upgraded immediately some of my cra projects to 1 to get the SW support. But cra is currently the entry point to react and in some cases to web dev for some. And I can totally can see how newbies can get confused. So I think we should make the simplicity the priority and make sws opt-in.
How about a "higher order package" create-reaect-pwa which would install CRA as dependency, forward all cli commands to it and simply inject the line needed for SWs when CRA is done?
I really appreciate @gaearon's openness to hearing community feedback about the pros and cons of keeping Service Worker support a default. It's been valuable hearing what Chrome and other browser vendors can do to improve our debuggability for PWAs. Ultimately, I'm supportive of whatever decision is made to ensure CRA continues to provide a predictable developer experience for users.
As the community shifts to offering better SW tooling across frameworks (we're working with Preact, Vue, Angular etc on this effort) we're aggregating specific work items we can tackle in both Chrome's SW implementation itself and Chrome DevTools. If we could come up with a set of browser items that we can work towards addressing here that would be appreciated. That way if/when SW are made an opt-in feature we can keep working together on improving DX to the point where they are less confusing.
@gaearon could you help @jeffposnick and I distill this list as you see it and we'll work on prioritization from our end?
Expanding on some of the comments from users:
What I like about Service Workers is they enable React apps to have resilience to transient network failures. If you're targeting mobile (a sizable number of Progressive Web Apps are built with React these days, including Twitter) this enables you to be able to use your app on the go and still deliver your users something meaningful when they hit a spotty network connection on a bus, in a coffee shop or while traveling. It's a nice to have if you're targeting users beyond desktop.
In Chrome, if you have a Service Worker registered and have cached scripts (like your React vendor bundle and your application code), we'll also early opt you into V8's code caching which can have a significant impact on the amount of React library code we need to parse and compile. Effectively, we can skip compiling from scratch and execute React app code much sooner. This means users can interact with your UI sooner than they otherwise would have if you were strictly relying on HTTP caching.
Are these things CRA should care about offering by default? Perhaps, but I'll defer to the project as y'all know what is most useful to users. I'd love to do what we can (fix bugs, more UX explorations etc) to make the overall PWA experience as stellar as we can even if that takes a while.
Iâm a bit overwhelmed by some other work right now so I canât compose a good replyâbut I think @ro-savage did a great job at outlining the issues, and it would be great if you could collaborate with him on creating a followup issue list. Iâm happy to review and provide feedback on it, but I wonât find the time to write it myself in the coming days.
service worker
for PWA!
The fact that CRA has built-in service workers support was an important point why I choose CRA for scaffolding (even that I hate webpack).
But like most folks starting with PWA development I have struggled with some gotchas (my testing server was adding cache control headers to service worker file - debug that ;) ) before I understood how things are and I can understand frustration with new technology.
I believe optimal solution is to show some user notification as suggested https://github.com/facebookincubator/create-react-app/issues/2398#issuecomment-304700498 by default that could be switched off by a config option.
Similar to projects like https://www.npmjs.com/package/create-react-app-sass we could also have a create-react-pwapp
to let people pick SW explicitly.
I think that @mpj highlighted in his comment above the issues with adding a plugin structure to CRA
I see two issues:
Issue 2 is more complex because of the (mostly) ambiguous comments about SW, which could be summarized as: "I'm glad SW is there but it caused problems".
Rewinding to the point of deciding whether to include SW (with our knowledge now) I would argue that neither "yes" nor "no" would have been the correct answer. Including SW added value to CRA for me, it was in fact a teaching moment, but one I wish I could have chosen. So +1. But the time and energy required on all sides has been excessive.
Proposal: there should be some mechanism that allows features like SW to be added to CRA. Such a thing should NOT violate the "No Configuration" dictum but I suspect there are ways to do this. I am thinking ENV=EXPERIMENTAL_ALL npm run build
and ENV=EXPERIMENTAL_SW
kind of thing.
This is just a straw man (straw dog?) proposal, there are probably better ones.
Again, thanks to all for the constructive conversation and for CRA.
PWA should be forced as standard and service workers
as part of it should be default in CRA. I think that now after service workers
are default in CRA a lot of developers got in touch with it or hear about it for the first time and just that (educating more people) should be enough to leave it as default. As developer you should have nerves to handle frustrating problems and service workers
can be frustrating no mater if you have experience with them or not. If you as developer have days that are not frustrating you should ask yourself if you are working hard enough đ
Yes there are issues with it but with more and more people on the same boat we will fix them together faster and I'm sure that more issues can be found if more developers use service workers
. It would also make pressure on Chrome and Firebase devs to fix their problems with it.
I'm using Firebase and it still has some problems with service-workers
but the pros are way more than cons. My suggestion would be to add a comment above the service worker registration that describes a little what the registration line does and that you can comment it away if you don't want to use service workers
.
I think that there are much more people that like service workers
to be default but the other part having problems makes just more noise đ so let us scream "leave it as default" đ
There are many ways to implement service workers
, and several different scenarios in which they can be applicable. This is why I don't think they should be enabled by default, but rather an option. There are some great tools already available, and can easily be added to your project.
Service worker caching should be considered a progressive enhancement. If you follow the model of
conditionally registering a service worker only if it's supported (determined by
if('serviceWorker' in navigator)
), you'll get offline support on browsers with service workers and
on browsers that don't support service workers, the offline-specific code will never be called.
There's no overhead/breakage for older browsers if you add sw-precache
to your build.
All resources that are precached will be fetched by a service worker running in a separate
thread as soon as the service worker is installed. You should be judicious in what you list in the
dynamicUrlToDependencies
and staticFileGlobs
options, since listing files that are non-essential
(large images that are not shown on every page, for instance) will result in browsers downloading
more data than is strictly necessary.
Precaching doesn't make sense for all types of resources (see the previous
point). Other caching strategies, like those outlined in the Offline Cookbook, can be used in
conjunction with sw-precache
to provide the best experience for your users. If
you do implement additional caching logic, put the code in a separate JavaScript
file and include it using the importScripts()
method.
sw-precache
uses a cache-first strategy, which results in a copy of
any cached content being returned without consulting the network. A useful
pattern to adopt with this strategy is to display a toast/alert to your users
when there's new content available, and give them an opportunity to reload the
page to pick up that new content (which the service worker will have added to
the cache, and will be available at the next page load). The sample service-worker-registration.js
file illustrates the service worker lifecycle event you can listen for to trigger this message.
It's very important that the requests sw-precache
makes to populate your cache
result in the most up-to-date version of a resource at a given URL. Requests
that are fulfilled with out-of-date responses (like those found in your
browser's HTTP cache) can end up being read from the service worker's cache
indefinitely. Jake Archibald's blog post provides more context about this problem.
Here are a few the projects I'm watching:
@addyosmani
Here is what I think is an emblematic example of Chrome usability issues.
I am running a file server on the same port as a React app which I last touched two weeks ago. React app pops up.
I Shift+Cmd+R for it to go away. It does. Then I click on another link on the same host and port, and bamâit's back again! I Shift+Cmd+R again. It goes away.
I click another link. Shift+Cmd+R yet again đĽ .
The last screen is especially funny (I have a bunch of iframes I need to test, and none of them show the right thing).
As a consumer I expect Shift+Cmd+R to wipe out the cache completely and delete the service worker. This combination is my last resort when things are broken. Since service worker "takes over" the whole website then I expect Shift+Cmd+R to be more aggressive in this case.
In my opinion tucking it anywhere into DevTools is a non-solution. It's not accessible to the end user, and even as a developer, I won't bother either toggling incognito or diving into DevTools when all I want is to view the website I'm interacting with now (rather than the one that was cached weeks ago).
Regarding https://github.com/facebookincubator/create-react-app/issues/2398#issuecomment-308106499, I've restarted a discussion on the Service Worker Specification issue tracker about formalizing a procedure for unregistering the old service worker in the scenario you describe: https://github.com/w3c/ServiceWorker/issues/204#issuecomment-306941887
Specs take time to change, and even if they're updated, it takes time for browsers to match those changes. In the meantime, the modification to @ro-savage's PR I describe in https://github.com/facebookincubator/create-react-app/pull/2438#issuecomment-306938882 would allow c-r-a to work around the scenario you describe immediately.
As to the behavior of Shift+CMD+R, the current behavior you're seeingâtemporarily disabling the service worker, but not unregistering itâmatches what's described in the service worker specification. Changing that temporary bypass into a permanent unregistration would have to go through the spec process as well, but perhaps it wouldn't be seen as necessary if they other changes are implemented.
My two cents: having this as default in create react app is causing issues we did a DNS cutover to another site. Users who ended up with service-worker.js are now "stuck" on the old site because the worker is attempting to hit the new DNS at /service-worker.js and the file doesn't exist, so it continues to show the old site.
Hitting Control-F5 brings up the new site, but hitting F5 brings up the old site again.
I just don't get why someone would want this behaviour - caching an entire site in the browser storage and continuing to show the cache if the currently site does not have service-worker.js
I could be completely misunderstanding how this works.
@eric-tucker If this is a problem right now you can explicitly unregister it, as documented:
If you had previously enabled service workers in your production deployment and have decided that you would like to disable them for all your existing users, you can swap out the call to
serviceWorkerRegistration.register()
insrc/index.js
with a call toserviceWorkerRegistration.unregister()
. After the user visits a page that hasserviceWorkerRegistration.unregister()
, the service worker will be uninstalled.
@gaearon That's not clear. I don't have src/index.js
Let's pretend the new site is absolutely no react (it's Drupal).
How does one solve this?
@eric-tucker I think you might want to paste this in your js:
if ('serviceWorker' in navigator) {
navigator.serviceWorker.ready.then(registration => {
registration.unregister();
});
}
Your trapped users will still need to hard refresh, but at least not-trapped users will have their serviceWorker removed.
Some want this behavior because they prefer a fast second load over a fresh second load. E.g. think of a page you might visit 100 times a day but only changes once every few days like the front page of facebook or google.
Personally, I'm on this github issue because I've been burned just like the gif @gaearon shared. My api is at example.com/api/bla
and the react app was hosted at example.com/
which caused download links to the api to show index.html
instead of the expected content.
If serviceWorkers stay I would request a route white-list and black-list to be included in registerServiceWorker.js
so this behavior can be configured. I think the fetch
event handler supports configuring the routes.
@ubershmekel users are stuck on the old version of the app with only requests to service-worker.js happening. Let's ignore the fact that default behaviour of the service worker created by create react app is caching index.html, there's only a single place I can attempt to hook into the running service worker to get it to stop, and that's in the code that refreshes itself.
So, with that in mind, is there any way to break the service worker from within its own code in service-worker.js (requests for this are being made at our Drupal site).
@eric-tucker, please see https://stackoverflow.com/a/38980776/385997 for an example of a service-worker.js
that you can deploy which will effectively act as a no-op.
Sorry for the slow response. I've been MIA.
I've now created a consolidated issues list #2554
Until most of these issues are addressed, I highly recommend cutting a release without service workers and releasing that ASAP.
To reiterate what's been said.
CRA should be easy and just work. Service workers are not easy and do not just work.
Let a react offline boilerplate handle service workers and figure it out the issues first.
CRA should have it optional and off by default or remove it completely.
@gaearon - if you let us know what you'd like to do, I am sure @jeffposnick or I could do a PR to remove or make it optional. If it's removed, we could always keep it on a seperate branch including the fixes until it's ready for prime time.
Thanks for creating that consolidated list of issues. That's helpful. Perhaps we should PR making SW an opt-in until we've worked through more of the items in that list.
Perhaps my only feedback on the consolidated issue is it could be useful to categorize the items there. E.g UX/DX/browser.
Deferring to @gaearon for insight on preferred next steps.
Having Service Workers and PWA implemented in next would put it in the hands of thousands of react developers who are using a more 'advanced' framework that already had a lot of features and people don't expect to 'just work' like CRA.
@addyosmani & @jeffposnick maybe you could chat with @arunoda about adding it in by default, if you'd like to get feedback from developers. I can see they already use service workers for pre-fetching and have examples on setting up a next app as a PWA.
based on this http://www.phpied.com/create-react-app-sw-precache-pwa/ sw-precache can be added on the user space without the need of fiddling with webpack config. And I remember a long time ago I have made PWA compatible CRA generated project with adding a post-build command. I can't remember whether it's sw-precache or sw-toolbox.
It's sw-precache! after digging in my old private repo commits I'm doing the exact same thing the tutorial did. The issue at that time is the same one that we have right now (caching, updating the service-worker.js). If I had known that I need to serve service-worker.js with no-cache header maybe the build script still in the repo right now. :)
So I'm voting for removing the webpack plugin, but just add a tutorial about sw-precache in CRA docs.
index.js
has the line serviceWorkerRegistration()
, not serviceWorkerRegistration.register()
.
To unregister
change
serviceWorkerRegistration()
to
unregister()
and change
import registerServiceWorker from './registerServiceWorker'
to
import { unregister } from './registerServiceWorker'
in index.js
I haven't read this whole thread so maybe this was covered already, but I found a case where service workers lead to confusing results.
My public/
folder contains two HTML files. One is the default index.html
, another is a small thanks.html
, which I want to serve directly. It's a small page that doesn't need React or any of the other machinery provided by CRA.
This works great at first. I can go to <my-page>/thanks.html
and see the file.
But after I load the main page, said URL no longer works. It loads the main page instead. (without changing the URL).
The simplest way to fix this was to opt out of service workers. That's a shame though, because they're great at everything else. It sounds almost like the caching CRA's default service worker performs doesn't account for some filetypes in public/
, and I don't know where to go to fix that.
Hello @SwizecâThe service worker that's generated as part of the c-r-a
build process will use the index.html
"fallback" for navigation requests if there isn't a match for a given URL in the precache list.
The easiest way to resolve the issue you describe is to ensure that your thanks.html
file is picked up in the list of resources that the service worker precaches.
I'm... actually not entirely sure what the best way to do that would be. The sw-precache-webpack-plugin
should precache anything that's in the assets list that Webpack generates as part of its build (excluding files that match staticFileGlobsIgnorePatterns
), so I think you'd want to figure out why your thanks.html
file isn't part of the Webpack assets?
(CC: @goldhand)
It's expected that it's not part of Webpack asset list because we manually copy any extra files in public
into the build folder before running Webpack (or maybe after). Please file a bug about this. I'm not sure what the right fix would be. We intentionally don't let Webpack know about those assets because there may be thousands of them.
@Swizec We can include assets that are not a part of the webpack asset list by adding two parameters to the SWPrecacheWebpack config:
mergeStaticsConfig=true
and staticFileGlobs=['public/*']
This only works if the assets are copied to public/
before the webpack compilation process.
Respectfully, it seems that the rationale for _not_ including redux, react-router, {insert-dependency-here} within CRA should also apply to service workers. Both are _very_ opinionated choices and aren't really core to the mission of creating _react_ applications quickly.
Having said that, I'm certainly not opposed to enhancing this project to including more robust functionality like redux support, generators, etc. But if that isn't the goal then I think service workers shouldn't be here as well.
2 cents.
Disclaimer: I have no idea how CRA works.
I'm not sure if this is again CRA ideology, but maybe it makes sense to have plugins/hooks which will allow usage of such things as serviceworker/redux/etc? I know the ideology is against any configs, but reading this issue gives an impression that it's possible to specify configs for some webpack plugins.
Wanted to pitch in and say, this needs higher visibility and some kind of informational text in install/readme.
I just spent quite some time debugging an app that cannot be cached for offline use which uses express session auth. While express was redirecting all requests to /login, only fetch requests to dynamic endpoints would get redirected and even after forcefully calling window.location.replace once a fetch was redirected, service worker would still cache the page load and keep the browser from getting redirected to login.
As @gaearon mentioned, this is a very powerful feature and I understand there could be need for it (offline use and aggressive caching) but this debugging experience left a very bad impression of service workers and CRA on me.
I'm having the same problem as sunsetfantastic in this reddit post. I didn't comment out the 'registerServiceWorker' code before running my app a few times and now I can't get ports 3000, 3001, or 3002 open. I've tried many suggestions including killing the process that was using the port from the terminal, unregistering service workers from Google DevTools, these steps, etc.
I've been trying to fix this issue on and off for about a week now with the help of multiple other people to no avail. Like sunsetfantastic I'm about ready to just reset my laptop.
I apologize if the actual fix is already in this thread but I couldn't find it. I read over half of the comments, but I did not read every single one because it's quite a long thread. Can someone please point me to a solution for opening up these ports and getting react off of them? I would hate to have to do something more drastic to clear this up :(
@meghanprestemon - You need to remove serviceWorkerRegistration.register()
and add serviceWorkerRegistration.unregister()
to your CRA app.
Once its unregistered on all your ports, you can remove serviceWorkerRegistration.unregister()
as well
However, if your site is live on the internet, you'll need to publish it with the unregister and leave it there.
Alternatively, if you use chrome you can delete the service worker. Open dev tools, select Application
choose Service Workers
, check Show all
. Then hit unregister on all the localhost urls.
Is it just me or did Facebook itself just run into this problem??? My Faceboook home page was coming back server error unless I hard refreshed, so I had to unregister the service worker and now it works fine again. Food for thought haha
This issue has been open since May . I was wondering if there was any status update on how and when this will be resolved?
I very much like the addition of ServiceWorker. I think it is an opinionated choice about the web as a platform and the api's we should support. However, I do not see it as a developer framework choice, such as redux, relay, apollo, etc. Just like React has to target certain browser versions, defaulting to have offline support with service workers is a good, forward looking decision that hopefully will push the web forward. I also love that it is the default, but is super easy to remove - just delete 1 line in the index.js file!
@ro-savage Thank you for getting back to me. As noted above, I have deleted ALL service workers through Chrome Dev Tools and that did not work.
Also, can you point me to where the serviceWorkerRegistration.register()
method is? I have the following two lines of code in the index.js:
import registerServiceWorker from './registerServiceWorker';
.
.
.
registerServiceWorker();
which I have changed to
import { unregister } from './registerServiceWorker';
.
.
.
unregister();
and that doesn't work. I don't specifically see the method serviceWorkerRegistration.register()
referenced anywhere in the index.js though. Should I be looking somewhere else?
I just wanted to check in again and see if there are any new solutions for this problem. I am still unable to get those ports free. Thank you
@meghanprestemon Very sorry you're having issues. I pinged @jeffposnick and hopefully he can help you out.
@meghanprestemon It would help if you could share your project, or at least the full content of your src/index.js
and src/registerServiceWorker.js
.
Hello @meghanprestemonâapologies for the delayed response. (I'm currently traveling, with less access to GitHub than usual, but I'll check in again for your responses.)
First off, it would be great to know what version of c-r-a
you're using. There are some mitigations that were meant to prevent the official development server from running into this issue introduced in https://github.com/facebookincubator/create-react-app/pull/2276, which should have been included in the 1.0.6
(or maybe 1.0.7
?) release. I'd like to understand whether you're running into this because you're on an older version of c-r-a
, or whether the changes in that PR still leave you in a confusing state.
Second, while it shouldn't be necessary to manually delete anything with the latest codebase, I'm trying to understand how you could be in a situation in which explicitly deleting the service worker registrations in Chrome DevTools could have left a service worker active. Have you quit Chrome in between deleting the registrations and trying again? Can you post a screenshot of your Applications DevTools panel, with Service Workers selected and "Show all" checked? It should look empty if you've deleted everything, like:
And finally, could you try opening a Chrome Incognito window and then visiting http://localhost:3000? There's guaranteed not to be any pre-existing service worker registrations in place when you open a Chrome Incognito window for the first time in a session, and if you somehow see content at http://localhost:3000 from an Incognito window, then it's due to a "phantom" web server instance running on your computer, and not due to a service worker.
You probably know that service worker won't work on any iOS device (20% - 25% of all your users) and this is not about to change soon (maybe never).
Just a small update. Webkit will support ServiceWorkers. It is in development.
Check isserviceworkerready for the latest news.
So I know I am not adding anything new to this thread, but wanted to comment that I recently moved our web app from a VS2017 template to separate back end (web API) and front end, using create-react-app. The first build to staging wroked great but the a second build seemed to not have gone through. My coworker told me it worked fine for him, so I hit shift+F5 (windows user here) and it refreshed and worked as expected. I suppose its a PWA thing, which i think my app has no use for but I am not against of. However, I recall from learning the basics of Webpack that you could create hashes on file names automatically, allowing the browser to use cached files but use newer when available. Can CRA use this feature and would this fix or mitigate the issue? In case it's not obvious, I understand very little of what I am talking about...
@alebrozzoSP Unfortunately that won't work for service works / PWA.
Webpack is still creating those hashes, but the service worker will load in whatever was last used/saved.
It won't do a check against the server before loading the site (thus why you get instant load, it parsing and rendering before it even attempts to talk to the server).
Once it finds the that the file has changed / new hash. It will then ask you if you'd like to reload the page.
Am usually a vanilla JS guy, so a total noob to React. I needed a service worker for a new project, so I'm so glad it's already there. Works as expected. Keep it.
Thank you so much @jantimon for opening this issue. You saved my afternoon. After hours of invalidating CloudFront cache, fiddling with S3 settings, clearing browser cache, testing in incognito and banging my head against the wall trying to understand why "curl https://my_url/callback.html" gives a completely different result than requesting the same URL in the browser with DevTools open + cache disabled, I finally stumbled upon #2398. I had never heard of ServiceWorkers before. Great solution for off-line apps, terrible idea to have it "on" by default.
Hello all, new user here and I just wanted to give you my point of view on this topic.
While getting started with CRA, I wish I hadn't needed to spend half a day researching why service-worker.js
, manifest.json
and asset-manifest.json
were magically appearing in my build and why I needed them at all. Everything else going on in CRA was extremely straightforward and including this by default seems to go against that spirit.
After learning about PWAs and service workers, I decided that these features are not worth the additional complexity for most of my projects. I also generally prefer to start simple and add features as I need them and not the other way around.
So, not only did I have to spend my time learning about something which is largely useless to me at the beginning of my project but I also incurred some technical debt in the form of my new build script (shown below) and even more time will be lost explaining all of this to junior developers who will take over the project in the future. I now also have to worry if CRA is going to magically put other things in my build in the future.
For now, here is my solution to disabling all of these things:
registerServiceWorker
from index.js as mentioned earlier.registerServiceWorker.js
from the project."build": "react-scripts build && rimraf ./build/service-worker.js ./build/manifest.json ./build/asset-manifest.json"
.I did not delete public/manifest.json
from my project because I'm not sure if CRA uses that file in any other capacity. Everything works if I do delete it, so probably not. I will have to revisit this in the future as I learn more.
@waynebloss the only required step is to remove registerServiceWorker()
from index.js
.
@Timer thank you, I did that but I outlined some other concerns that are not solved by that action. Specifically, that ./build/service-worker.js
is still being generated. So, that needs to be addressed somewhere along my deployment pipeline because I'm not going to deploy files that are completely unnecessary.
FWIW, this manifested itself as mysterious behavior and almost ruined a demo to my client. It was masking my login endpoint by serving up the index page. I'm still not quite sure how it did that. I don't need the speedup or offline capabilities, and definitely not at the cost of mysterious behavior. That it disables itself until it hits production makes its behavior even more surprising.
I'm all in favor of fast, offline capable webapps, but this feels like premature optimization.
Here's one possible change that could be made short of removing service worker generation: by default, index.html
is used to handle all navigation requests for URLs that aren't precached (config).
This behavior is in there to support SPA-style offline usage, where users might navigate to URLs that don't correspond to underlying HTML files. That being said, many developers don't fall into that category.
What we could do is remove navigateFallback
/navigateFallbackWhitelist
completely from the default configuration. Offline usage for index.html
and any other files that exist as part of the Webpack build will still work. Offline usage for SPA-style routing will continue to work if hash
-based routing is used (as opposed to pushState
routing).
The drawback is that getting offline support with pushState
routing working would require an eject
and re-adding navigateFallback
to the config, but at least there's a specific place in the existing documentation for this use case, where we can clearly spell out the changes that are needed.
This would address #3008 as well.
OK, Newbie here to react. This file although made me start looking into an entire area of caching that I hadn't before; I don't think should be enabled by default. It just ruins the experience of a quick hello world hack where you're trying to get something running with minimal code and makes you research for hours on areas that I don't necessarily need right away.
I use create-react-app frequently and I contributed to this thread earlier when I "stumbled" into service workers by accident. I'm invested in CRA continuing to be a good tool that is helpful for people. As I've watched this thread it seems like there are many pro SW posts and many anti SW posts.
Resolution then depends not on votes but on the purpose of CRA. Is it something for newbies? If so then adding things like SW is a definite minus. One could think of that doctor thing where the first rule is "do no harm." As many people have pointed out they have experienced at least pain from the inclusion of SW. So if this is one product and that one new product is designed to bring new users to React, then it appears this should be an easy decision.
If that is not the case then the target audience needs to be documented, "Not for new users", and leave it in. I can't see this. If someone knows enough to use SW and what the gotcha's are then it should be trivial to uncomment a line in index.js or App.js ?
React is very good. CRA is very good - it has saved me untold hours of fiddling with and needing to know a large number of build technologies, so it would be nice to not have this kind of thing happen again.
Personally speaking i found it trivial to just remove the one line from my index.js and delete the js class file.. however having an option in the create app wizard to include this would be helpful to those who might not understand what the SW is or how to disable it.
Just to be clear, if this is a tool targeting newbie users, I am proposing everything is the same, except for the one SW line in index.js is commented out. The documentation is change so there is an "Adding SW" section along with "Adding Bootstrap"et al. "Targeting" in this case means not exclusively new users, but the case where new users are an significant part of the target audience.
Just to be clear, if this is a tool targeting newbie users
no it is not. It targets all users. It is easy to use for newbies (because there is no milion configs, instead good conventions). It allows to use advanced features as splitting js in chunks and lazy load or use Service Workers. I do not feel it is right to treat it as newbie or advanced tool. This is just a good tool (toolbox in terms of Dan Abramov).
I am proposing everything is the same, except for the one SW line in index.js is commented out. The documentation is change so there is an "Adding SW" section along with "Adding Bootstrap"et al.
Yes please.
And the reason is not newbie users, but because it is not safe to enable this feature unless you can control Cache Headers for service-worker.js
- which is out of responsibility of c-r-a. Instruction should explicitly say: enable this feature only if you can setup CacheControl: 'max-age=0'
for service-worker.js
and also encourage to change console.log
to something more user friendly in registerServiceWorker.js
.
So, it seems like the majority here feels that this feature should be opt-in.
Besides that, this is an experimental feature that is only supported by 2 browsers.
What are we waiting for?
Besides that, this is an experimental feature that is only supported by 2 browsers.
This is incorrect information.
@NekR That is correct according to the top results that I get when searching for service worker browser support. This Mozilla page says it's experimental. This page, which Google points to as the place to check says that Chrome and Firefox have shipped it.
Can you show me some conflicting information? Thanks.
ServiceWorker is in development in both Safari and Edge. It has been shipped behind a flag in Edge 16.
"Experimental" on MDN is computed automatically based on the table of supported browsers, I believe. It doesn't take "in development" status into account though.
Here some more info about SW status: https://jakearchibald.github.io/isserviceworkerready/
Given all that, ServiceWorker technology is not experimental.
@NekR you have a horse in the race. To me - any technology that isn't fully supported by all browsers for at least a whole year is experimental. MDN defines service workers as experimental. "In development" also implies experimental. You can have your own definition and opinion for "experimental" but your statements read as wrong, misleading, and borderline rude to me.
@ubershmekel you're only rude here. First of all, this is not a race. My open source ServiceWorker development has nothing to do this issue. I'm not implying someone should use my work, stop using current work here, or stop/continue using ServiceWorker here at all.
Second, I'll tell you why there it is no support "by all browsers for at least a whole year". Because it takes years to implement ServiceWorker support, both mentally and in coding hours. ServiceWorker is one year in development in Edge by now and was the same or even more in Chrome.
Third, process of "in development" doesn't imply "experimental" by any means. All vendors agreed to implement the technology as it's now. All are committed, as well as major website players, e.g. Facebook who is participating in ServiceWorker spec development.
Fourth, if you really hate experimental features and believe that all vendors must implement them to take them out of experimental, you may want raise issues about babel-plugin-syntax-dynamic-import
and babel-plugin-transform-class-properties
syntaxes use in this project as well.
Anyway, if you wish ServiceWorker support removed from this project, that's your right. But argument about ServiceWorker being "experimental" is 100% incorrect.
you may want raise issues about babel-plugin-syntax-dynamic-import and babel-plugin-transform-class-properties syntaxes use in this project as well.
While I don't dispute the rest of your post, having grown detached from the discussion with time, I'd like to point out that this is not the same. Using service workers relies on browser support. Babel plugins transpile proposal code to something supported by all relevant browsers, which is the point of using Babel. AFAIK there's no babel-plugin-service-worker.
@NekR Call it whatever you want. I would prefer to not have "experimental" or what you consider "in development" features in my production tools, thanks! Obviously many people here feel the same way.
@alebrozzoSP How exactly do you plan to consider the community as a whole then? The way I see it, the people from the community that care about this issue are already here.
TL;DR I am absolutely floored. I _hate_ this being on by default. Had me hunting a ghost. My vote is that this absolutely cannot be on by default. My suggestion is to add a documentation header in the registerServiceWorker.js directing people to the readme.md for details of use. But do not make this on by default until it is supported by all modern browsers -- I consider Safari a modern browser.
I develop on a Mac. I test on an iPhone, and my Mac during most Unit Testing scenarios, and only get Windows and Android involved when I am done with my initial Unit Testing. I learned C-R-A before this service working thing was added. I have a Firebase project that for the life of me, I could not get to load. I spent all night yesterday trying to figure out why my deployed code was not working. I learned I had to delete the website data for my site to get it to work. But then after a short period of time, nothing. my browsers would go out to lunch.
On my Mac, I discovered that Chrome browser worked, just not Safari. So, I decided to download Chrome Browser for iOS, and tried it there. It worked there too. Now I was stumped. React has always been "all-platforms" compatible in my experience.
Now I know why my app stops rendering at random times, or just goes out to lunch completely after deploying new code.
Who knew....?
## Update: If the end user of the application is to be considered, consider this: registerServiceWorker does not work on Internet Explorer 11, Mac Safari, & iOS Safari. There is no question that Internet Explorer 11 (and prior) is the number 1 deployed browser on the desktop, and in corporations. It may not be you we use, but it is what most people use. Even of the system with Windows 10 + Edge, I do not know anyone to uses Edge. If for no other reason, because IE is what they know.
It's sad and absurd that service worker code was dumped into this project. CRA should be as simple and clean as possible! Service workers are things that come _way down the road_ for any project, after business consideration and research.
hello , i don't find registerServiceWorker in my index.js, i've tried to read all your messages, i'm not against the new, i like the service worker idea, but my main problem is , why when i create a new code, deleting all the files of my old build folder, and copy the new built files, my chrome keep showing me an old version, if a new main.xx.js is genereted for every file, how to change the servie worker or something to automatically forget the old and charge the new version. I can clean form my browser, but I can't do it in all chromes from another users , maybe I am doing something wrong about how to build, and replace old build files .. ? I've read the docs pointed in the discussion but nothing clear about how that can work for all users without do it manaully
@webmobiles The service worker uses a cache-first strategy, so on visit N you'll see the previous version of your web app, and then the new version will be downloaded in the background, and on visit N+1 you'll see the updated version.
If you've removed the call to registerServiceWorker()
from https://github.com/facebookincubator/create-react-app/blob/master/packages/react-scripts/template/src/index.js#L8 then no one who uses your app for the first time will end up getting a service worker installed.
If you've previously deployed your web app with a service worker, and you've changed your mind and want to remove it from users' browsers, then call unregister()
(exported by the registerServiceWorker.js
module) from index.js
instead, and rebuild and redeploy your web app.
There's more info about both of these topics at https://github.com/facebookincubator/create-react-app/tree/master/packages/react-scripts/template#making-a-progressive-web-app
thanks @jeffposnick , I'd like to confirm the right code with the right paths on my project, the docs says:
If you would prefer not to enable service workers prior to your initial production deployment, then remove the call to registerServiceWorker() from src/index.js.
If you had previously enabled service workers in your production deployment and have decided that you would like to disable them for all your existing users, you can swap out the call to registerServiceWorker() in src/index.js first by modifying the service worker import:
import { unregister } from './registerServiceWorker';'
but the correct code is not this ?
remove from:
/node_modules/react-scripts/template/index.js on your project
and If you had previously enabled service ....
import { unregister } from './node_modules/react-scripts/template/src/registerServiceWorker';
unregister();
or am I wrong ? and every time I update react-scripts package must to change by hand ?
@webmobiles Removing registerServiceWorker()
initially will make sure that the service worker never gets registered/installed.
If you've previously shipped your web app with service worker registration enabled, and you'd like to remove it from the browsers of users who have previously visited your web app, taking the step of explicitly adding the call to unregister()
and redeploying your web app is the recommended course of action. Because the service worker serves navigation responses cache-first, the next time a user visits your web app the unregister()
won't take effect, but then the visit after that, it will take effect.
These changes would be made in your project's local index.js
file, and I don't believe that the local index.js
gets overwritten as part of react-scripts
updates. (But correct me if I'm wrong?)
@jeffposnick I understand now, maybe i was using an old version of create-react- app or react-scripts or I've paste index.js content from another project with those lines... so now I'm creating a new project and moving the files to the new structure where I see those lines in my index.js
the strange thing it's that build was working with workers, and that got me crazy, because i did not found those lines on my index.js
thanks for help me, now I can delete that line with updated packages and i think is going to work,
by the moment i don't want to use workers because I keep changing version for users and tell them, look at the new version, they confused because they get -1 version and it's late if they see the new version after , but i will use again when i have a stable verison of my app )
I'm not sure if this is the right place to talk about this, but the default manifest.json does not include a 144x144 icon image. If you include that in the manifest.json, android devices will prompt the user to add the website to their homescreen.
It was pretty hard for me to get that to happen on my own, but you guys are 99% of the way there.
@jeffposnick And if you have instead pushed a version first with serviceworker, then removed the whole registerServiceWorker.js
and registerServiceWorker()
WITHOUT first using unregister()
you are stuck with the old version for about a week. Sure with desktop you can purge your cache but on mobile it's not so easy and _very_ annoying. Not something I'd want for a beginner (or just who is unfamiliar with how serviceworkers work) to face when bootstrapping their first React-app.
Had this page on my tabs for weeks so I thought I'd finally contribute. But the previous was and is my biggest gripe with how this serviceworker is configured. Possibly a source of huge annoyment with little benefit. It should be optional and intended only when the app is beyond prototyping and optimizing the pageload would be beneficial. Like I'd want create-react-app to have modular addons so I could add say react-redux/serviceworker/etc with just npm run add <feature>
and keep the basis as simple as possible. Without serviceworker. đ
@TeemuKoivisto Here is an article about properly removing ServiceWorker: https://medium.com/@nekrtemplar/self-destroying-serviceworker-73d62921d717
But I guess problem is in how hard it's to do in CRA. Do you think it makes sense to have cra-remove-serviceworker
module as it has gatsby-plugin-remove-serviceworker
?
I haven't had a terrible experience with Service Workers, but I believe that's only because I'm reading this thread BEFORE actually putting my app into production.
I'm gonna disable it and here is why:
You want to educate people who use RCA? Don't make it a default, make obvious console message about what they can gain by enabling it, links to proper explanations and tutorials. Devs will notice but won't lose hours/days trying to figure something out that is so hard to trace/debug.
That technology, used properly, has definitely huge advantages. Misused, it has huge consequences. I'm gonna stick with the normal/standard/toBeExpected behavior of my websites/apps for now. I'll dive into it and run proper tests and experimentation when I actually have the time&interest to do so.
My take on this is that it simply does not belong in Create React App. CRA is about creating a react app. The Service Worker has absolutely nothing to do with react. the registerServiceWorker.js should be node module to import by users that want it. All the things in CRA are actually used to develop and deploy your first react app. the service worker is not used for anything, but leveraged by some browsers (not all). Our organization has standardized on turning this off by default, and I have not spoken with a single developer yet that thinks this is a good thing for CRA. I find it very frustrating that this even needs to be a discussion. the registerServiceWorker is as appropriate for CRA as forcing React-Bootstrap or Material-UI into CRA. It is not aligned with the CRA goals as I see it.
Reading through this thread, it was recommended in June to cut a release without service workers, and PR #2995 has been open since August to disable service workers by default.
What exactly then is holding create-react-app back from making service workers opt-in instead of opt-out?
Our team has been working on a web app since September and using an incremental release strategy. This means that every week or two we are sending to prod a new version of the app. Every time users go and want to see the announced new features, they complain that they cannot see it. Same thing with user acceptance tests. It is most problematic when the features added are breaking changes (like an API modification or something). We are considering removing the service worker altogether but reviewing this issue thread I just saw about the toast and callback options as posible solutions, both of which I like... possibly a callback function with a default display of a toast that devs can change to either restyle or force a refresh of the page or whatever the need.
Is there a definition on CRA maintainers of what the direction will be? a tentative roadmap maybe? I think SW enabled by default are ok as long as both devs and users can tell when they can/need to refresh the app.
I recently ran into problems where the service worker was trying to fetch outdated dynamic import/code split pages which resulted in my app crashing when navigating to a new page. See #3613
Please disable the SW, because when the devs aren't aware of it and deploy on CDN like Firebase, it f**k us up so hard, and took hours for us to try to get user's browser to reload the new js/pages.
I don't want to be rude, but we're so fed up with it.
@smiled0g how can you deploy a application without noticing the registerServiceWorker()
part in the index.js
file. And if you notice it, are you not curious what that code is doing in your project? It is not hidden or uglyfied or somehow else hiding from you?
I don't want to be rude, but people are investing so much time and afford in this project to be up to date and to support all the standards every web application should have. Everyone who is using it should at least invest some time reading the docs before complaining around.
Maybe we should take a moment to read through this whole issue full of comments from people who encountered problems because they did not expect something called create-react-app, not create-react-app-with-service-workers, to create a react app with service workers.
Regardless, it's obvious the maintainers don't want to change this. There's a clean and simple open source solution for this issue. If you care enough, create a fork, disable the service workers in it, publish it as a different package. If people start using it instead of CRA it will send a clear message that majority of devs don't want service workers enabled by default. If not, then it wasn't that big of an issue anyway. Open source is magic.
The service worker is stupid. Its loke buying a car that runs on gas. Putting gas in it, and after the fact learning that once you put gas in the car, you are now forever stuck with that type of fuel. The service worker should be itâs own technology that is not part of CRA, it makes no sense to include it at all. And it definitely should not be enabled by default.
After watching this issue for a long time, my take away is to question the purpose of CRA. My thought was that it was a tool to help users with React. And it was great at that, partly because it was not opinionated.
When I run through the documentation with the list of things CRA does not do because they are not standard (see for example Decorators) and the list of optional things you can add in, Service Workers are a complete outlier. [ I'm open to other interpretations?]. Instead of "Here's how to add it" or "we don't support it because it is not standard" it is enabled by default.
Of all the things, about SW and CRA, that is the greatest concern I have. It would be nice for someone to explain, not why SW are great, or why they are terrible, but why their inclusion seems so contrary to everything else going on. Anyone?
I have great appreciation for the people who created CRA, it has made my life as a developer much better, so I am not against CRA or even SW ( I will probably use them at some point). I am concerned by the philosophy change [barring some explanation].
So I think the suggestion by @MilosRasic is the right answer. To fork CRA to CRA-Safe or something like that. If we have to run through each CRA release to see if it now contains some new hazardous material (attempt at humor) we might as well do it together.
I have mentioned it before, but I can mention it again. CRA works on all web browsers I have tried. Service workers dont work on Internet Explorer. It causes huge issues for any corporate devlopment environment as all application release updates requires a forceful cache flushing process that is not streaming VDI friendly. When using a streaming VDI, the cached content is stored in the users profile. You cannot just reset the cahed data, you have to log off and back on to force update the network profile. If any user signs in from another computer before signing out, then their profile is not ipdated.
There is no good way to force a clean deployment. Too much development today assumes the world lives on a single, non-corporate attached computer, and fail to deliver a true enterprise friendly solution.
We standardized on CRA before SW was added. We spent hours trying to figure out what happened to CRA apps.
Adding new functionality like this, which is a 100% optional feature that is not a react solution, but a browser acceleration technology, and making it default.
The team might just want to enforce offline-first methodology, they even have it written in readme:
By default, it also includes a service worker so that your app loads from local cache on future visits.
If you don't agree with their moves, leave a rational comment, make your own fork, or just disable it yourself. Don't be too emotional or even rude here, they don't owe you anything.
If anyone's interested in having CRA with server side rendering AND nice offline experience with a service worker, here's my take: https://vfeskov.com/2017/12/29/Create-React-App-with-SSR-and-Service-Worker/
Basically I make my own service worker, since the default CRA one doesn't have any relevant options :(
I hate this addition too... So many people have complained they don't see the latest videos on reactu.io - I look and sure enough, it is stuck on some old cached version.
I also experienced a fair amount of confusion due to the service workers. For me it made it look like my nginx configuration was working properly because refreshing when on a react-router "page" caused the service worker to return that "page". When really the request should've been sent to the backend app and caused my app to blow up. You could see that happen when trying to go to a react router url out of the blue, without it being pre-cached by the service worker. I wasted many hours debugging that.
Okay, so today i decide to make a very simple app and add something like <meta property="og:image" content="%PUBLIC_URL%/hp.jpg" />
inside my index.html
.
It should just work and it did in local, but not in production.
After pulling my hair for 3 hour, i just found out that after you run npm run build
, you can not access image because it is now served by serviceWoker
which for some reason doesn't recognized that I want to access the image (not index.html
for f sake)
CRA should just work, service worker is not only magic but also buggy as hell
Create React App is the best way to start building a new React single page application.
I think the project could benefit from a definition of it's intended audience.
Because the React docs now promote CRA as the best way to get started _beginners_ are running into these issues which can take some time / research to resolve. If it's for beginners and should Just Workâ˘ď¸ then SW was merged prematurely.
If it's primary audience is experienced dev's or an educational tool to promote new tech like SW then it has a place here. I believe _this_ is more the intended audience for CRA, people look to it to see what an advanced build / testing setup looks like. They know when they start out what's included and what the gotcha's are.
I would disagree with the last statement. Even if it is for advanced users, SW does not belong as a default enabled state. Because it is not a mature sokution, does not work on a majority of enterprise class browsers, and it was added very very late in the game. With all other features of react, you opt-in to use them, by actually implementing them in code. There is nothing required to make your react app work with a minimalist setup, other than run CRA, and code. Woth service workers, you have to know to turn it off, and how to unload the cache from clients. It is a destructive on-by-default service, which breaks this paradigm of CRA (even under the situation that this is targeting advanced users). It simply does not belog. This is not a react feature, this is a browser implementation feature. It is no different conceptually than packaging a proxy server with CRA and enabling it on by default. React does not use SW, the browser does. And SW is not a feature of the app, it is a wrapper to manage client side caching. Nothing about service worker affects the react framework â nothing... whatâs next? Installing a pure javascript chrome browser to run react clients within, and forcing it in by default too? Where does the line get drawn? To me, it is pretty clear. If the feature is not a part of, or an enhancement to the react framework, then it should not be a part of CRA.
Heyaâ
We intentionally decided to ship this to see how this technology holds up in real world. I believe this gave browser vendors and people pushing for its adoption a chance to smooth out the rough edges from real world usage.
Itâs been almost a year now and think itâs fair to say enough time has passed that we need to make a decision. Before cutting the next major weâre going to revisit all the existing pain points, and figure out a plan. I think itâs fairly likely weâll disable it by default based on all this feedback. Ultimately we want whatâs best for our users, and if it causes more pain than it solves then it should be opt-in, not opt-out.
Actually I'm going to close this issue in favor of a more focused https://github.com/facebookincubator/create-react-app/issues/2554.
Weâre moving forward with making service-workers opt in in the next major release.
Theyâre still readily available behind a single line change but wonât be registered by default.
https://github.com/facebookincubator/create-react-app/pull/3817
I hope this is a satisfactory conclusion for most users. We can revisit this in a year or two.
That makes me get the warm and fuzzies, thanks!
Mark Tellez
Technology Consultant
marktellez.com
codementor.io/marktellez
FREE ReactJS Screencasts: https://www.youtube.com/channel/UCx4a8EMmXx-6RuJlyAKASoQ
On Jan 18, 2018, at 6:18 PM, Dan Abramov notifications@github.com wrote:
Weâre moving forward with making service-workers opt in in the next major release.
Theyâre still readily available behind a single command but wonât be registered by default.3817 https://github.com/facebookincubator/create-react-app/pull/3817
I hope this is a satisfactory conclusion for most users. We can revisit this in a year or two.
â
You are receiving this because you commented.
Reply to this email directly, view it on GitHub https://github.com/facebookincubator/create-react-app/issues/2398#issuecomment-358826045, or mute the thread https://github.com/notifications/unsubscribe-auth/AVchh1zk8AGw52EvnRzaxTbdQPPXUKWxks5tL99igaJpZM4Nordg.
as an electron developer whom uses CRA to generate my projects, SW is not used in these types of software applications. Removing SW is usually the first thing I do. having this opt in will save me countless minutes of having to remove that one line and file from the project ;) Anyways i think its a good paradigm to follow though, why add in stuff that should be optional. Thank you.
Good guys win again!
I'm using express to serve a dynamic index.html and the service worker was driving me crazy until I found this issue.
Thank this shit is gone.
I think it should come commented by default, with a comment saying something like "remove comment to enable this, etc".
This has already been implemented and will be released as part of react-scripts 2.0: https://github.com/facebook/create-react-app/pull/3817
Service workers are now opt-in and use Workbox.
https://reactjs.org/blog/2018/10/01/create-react-app-v2.html
Most helpful comment
I see your point but I don't agree removing a single line from
src/index.js
is a big deal (and that's all you need to do to opt out). If you can clarify what exactly makes your life harder (in practical terms, e.g. a specific scenario that is confusing) maybe we can make it better via other means.While iOS is lagging behind, there is more momentum than ever behind using service workers for webapps (basically everywhere except iOS), and weâre going to stick with this for a while. If people donât find it useful and most of the feedback we get repeats yours, we will remove it. So far the reception has been positive, but itâs hard to say until we live with this change for at least several months.
If the tool was called âcreate react websiteâ then Iâd agree with you, but people expect more responsiveness from apps, and precaching helps bridge that gap. Weâll see if it works out or not.