Most active commenters
  • dbushell(6)
  • kstrauser(5)
  • MartijnHols(5)
  • stavros(5)
  • (4)
  • echoangle(3)
  • kelvinjps10(3)

Et Tu, Grammarly?

(dbushell.com)
275 points dbushell | 79 comments | | HN request time: 2.847s | source | bottom
1. dbushell ◴[] No.43514309[source]
How do you deal with hostile browser extensions?
replies(5): >>43514670 #>>43515016 #>>43515058 #>>43516270 #>>43516486 #
2. WJW ◴[] No.43514670[source]
Uninstall them?
replies(1): >>43514789 #
3. dbushell ◴[] No.43514789{3}[source]
If only I could opt-out, disable, or uninstall those used by visitors to my website when the extension breaks it :(
replies(1): >>43515010 #
4. ◴[] No.43514962[source]
5. Aldipower ◴[] No.43515000[source]
I've a similiar problem with Google Translate that breaks my web app. Users, using Google Translate, complaining my app is broken, but it was Google changing the state of my app from a higher meta level. Really bad practice..

I am trying to detect Google Translate and print a warning then.

replies(2): >>43515029 #>>43516979 #
6. diggan ◴[] No.43515010{4}[source]
Not much you can do, user agents continue to act as agents for the users, meaning you can serve them stuff but beyond that it's up to them to dictate their experience, for better or worse.

It really sucks when extensions do fudgy stuff in global space and sometimes break your stuff though, agree. Best approach I've found is to have a help page you can link to so people can go through the typical steps of "disabling all extensions, clearing cache, etc, etc" when things break in very unexpected way and you find no causes for it.

replies(1): >>43515107 #
7. netsharc ◴[] No.43515016[source]
I guess you could figure out valid states for your page's DOM, and a few seconds after the page has finished loading, scan it for "hostile" elements and CSS styles, and delete them...

Having this idea and opening a random page (from The Guardian) on DevTools, somehow somebody's inserted scripts and iframes pointing to twitter.com.

replies(1): >>43518857 #
8. netsharc ◴[] No.43515029[source]
Maybe related, from 2 days ago: https://www.pewresearch.org/decoded/2025/03/21/how-a-glitch-... / https://news.ycombinator.com/item?id=43441880
9. eadmund ◴[] No.43515058[source]
I don’t think that ‘hostile’ is really fair in this case, when ‘insufficiently competent’ will do (albeit at the cost of more syllables).

I am not a fan of Grammarly or their technical model, but I don think it’s fair to attribute malice when it is adequately explained by stupidity.

It’s been a long time since I did any front-end work: should both Grammarly’s extension and your own code use namespaced property names?

replies(4): >>43515091 #>>43516294 #>>43516464 #>>43516925 #
10. preinheimer ◴[] No.43515079[source]
My extension problem story is a bit different. We distribute an extension that makes it easy to switch between proxy servers for geolocation testing.

I ran my worst client demo ever a few months ago. It was like our product simply didn’t work. A lot of pulled hair and frustrating debugging later we discovered that a recent update to the 1Password extension broke ours. They were subscribing to an auth event, but not returning, this timed out so our subscriber was never called. So our extension would tell the browser to change proxy servers, then sit ready to provide credentials, but the request would never come. 1Password’s support team was better than grammerly’s, but it’s hard to convince an unknown PM to prioritize something, especially if you’re speaking to them via a support team.

We’ve since discovered that there’s some Russian extension you need for government websites that has the same issue.

replies(1): >>43516584 #
11. dbushell ◴[] No.43515091{3}[source]
yep, it's in Grammarly's interest to namespace or scope their CSS in a way that doesn't conflict. Not doing it adequately goes both ways, website CSS could break their extension, or their extension could break the website.
12. dbushell ◴[] No.43515107{5}[source]
Indeed. It's the user's browser and experience, it's not up to the website owner. But it's frustrating to get bug reports when a 3rd party extension is the problem, not the website. Many visitors will just bounce blaming the website too.
replies(1): >>43515786 #
13. jFriedensreich ◴[] No.43515125[source]
Its frightening to see how many screenshares and recordings contain that green infestation as default on every website, not just the obvious visual disturbance (am i the only one who thinks the green is ugly and clashes with most websites colors?) that does not seem to bother users but the privacy and obvious attack vectors that come with it. Chrome can enable extensions only when needed why does no one do this? Why is this not the default on every browser?
replies(2): >>43516038 #>>43517379 #
14. ziml77 ◴[] No.43515786{6}[source]
Without someone pointing the user to what the issue is, it's very difficult for the user to know it's an extension causing the problem.

Many years ago I had performance issues with a site and the only reason I knew it was due to an extension is I dug into it with the dev tools and managed to identify Dashlane as the problem.

15. vhantz ◴[] No.43516026[source]
Do you know how they managed to inject stylesheets into every page bypassing CSP?
replies(3): >>43516075 #>>43516320 #>>43516388 #
16. the__alchemist ◴[] No.43516038[source]
Clarify?
replies(2): >>43516348 #>>43516742 #
17. ◴[] No.43516075[source]
18. kstrauser ◴[] No.43516093[source]
I passed this along to the engineering team.
replies(2): >>43516920 #>>43517675 #
19. karaterobot ◴[] No.43516187[source]
If you're injecting scripts or styles into unknown pages, the least you can do is namespace your variables.
replies(4): >>43516604 #>>43516740 #>>43516957 #>>43517036 #
20. amelius ◴[] No.43516270[source]
At this point, I'm not installing any browser extensions, period.
replies(5): >>43516317 #>>43516454 #>>43516480 #>>43516755 #>>43518863 #
21. QuadmasterXLII ◴[] No.43516294{3}[source]
We tried applying cunningham’s law widely and it created disastrous incentives. It’s better to assume profitable yet destructive incompetence is malice.
22. gs17 ◴[] No.43516317{3}[source]
OP's problem wasn't that they had it installed, it's that enough of their users did to make it a problem when it breaks the site's CSS.
23. daquisu ◴[] No.43516320[source]
It is done by the extension without any fancy stuff. Extensions can load static js / css and bypass CSP with it, if it is declared in their manifest.json. Grammarly's manifest.json is here: https://gist.github.com/Daquisu/11eb1a7000b4141c4404edcc6e16...

For more advanced CSP bypass with extension, you can:

1. Inject JS code into any webpage with a CSP.

2. Create an event listener for your content script and reacting according to it.

3. Use your content script to communicate with the background script.

4. Use the background script to communicate with any website, including blocked websites by the CSP.

Basically, any website <-> extension content script <-> background script <-> any website.

24. regularjack ◴[] No.43516351[source]
Et toi
replies(2): >>43516447 #>>43516472 #
25. emptysea ◴[] No.43516362[source]
At work we have a lot of sentry errors related to browser extensions doing weird stuff.

Chrome’s Google translate is also notorious for breaking react based sites.

It ends up being a tedious triage process to ignore each new extension issue. We use the client side filtering to reduce our ingest volume. In general we have to have a lot higher thresholds to handle the noise vs our backends.

replies(2): >>43516812 #>>43516871 #
26. ◴[] No.43516388[source]
27. meesles ◴[] No.43516447[source]
https://en.wikipedia.org/wiki/Et_tu,_Brute%3F
28. bmacho ◴[] No.43516454{3}[source]
Browser extensions are the equivalent of running random .exe on your computer except that you have to trust every vendor protecting their keys forever due to the autoupdate.
29. chuckadams ◴[] No.43516464{3}[source]
Any sufficiently advanced stupidity is indistinguishable from malice.
30. xandrius ◴[] No.43516472[source]
"Et tu" is indeed correct.from "Et tu, Brute" by William Shakespeare.
31. bufferoverflow ◴[] No.43516480{3}[source]
But our users do.
32. Doctor_Fegg ◴[] No.43516486[source]
Ah, my favourite complaint for the community website I run. "I can't see any photos on the adverts page." Are you running an ad-blocker? "Yes." What do you think an ad-blocker does...
33. horsawlarway ◴[] No.43516584[source]
Yeah, similar boat here. 1pass still breaks opening the chrome sidepanel UI from content scripts in other extensions. They screw up the trusted flag that indicates the event is coming from a user interaction.

Ultimately, as someone in the extension space for more than a decade now, Google is really at fault... Manifest v3 is just crappier than it should be in a LOT of ways (entirely outside the politics of the ad blocker changes - which is a whole different can o worms).

Overall, I think the quality of the chromium codebase feels a lot lower than it used to.

34. tikhonj ◴[] No.43516604[source]
Hell, namespacing makes life easier even just for yourself. I wrote some browser automation in a previous role that was never going to be user-facing—it wasn't an extension—and it still proved useful to namespace things, both to clearly mark what we were inserting vs what was already there, and to avoid possible collisions.
35. bryanrasmussen ◴[] No.43516740[source]
this really pisses me off because about 5-6 months ago I was doing an interview for a job that of course I did not get because old, and I talked about an instagram / branding startup thing I was the CTO of and main programmer in 2014 and how I made this build system to make sure that css classes and JavaScript objects were properly namespaced and how we made sure there was no potential collisions and the way we made sure exactly what scripts needed to be loaded on the page based on which of our widgets were on the 3rd party client site etc. etc. and at the end of it the guy interviewing me said dismissively there are tools that do that and everybody does that nowadays which I sort of had to agree they probably did because who knows, I'm not really doing that thing any more, and now it turns out they don't even.

on edit: fixed some grammar

replies(2): >>43517008 #>>43517018 #
36. echelon ◴[] No.43516742{3}[source]
I suppose they're concerned so many people are blindly installing Grammarly without a sandbox.

I'm concerned too, but from the angle that writing on the internet is becoming less human, more robot protocol. Even when it's from humans.

As if bots weren't enough of a problem, imagine when social media is just people clicking on buttons: "write a funny response", "write a comment in disagreement", "write 'same'", etc.

replies(1): >>43518609 #
37. silvestrov ◴[] No.43516755{3}[source]
The biggest problem with browser extensions is that the source code (both css and javascript) is not easy to read/check.

There should be an easy "view source" for extensions inside Chrome and extensions should be mandated to ship non-minimized code.

38. jgalt212 ◴[] No.43516792[source]

  - Access your data for all websites
  - Display notifications to you
  - Access browser tabs

> They could also, you know, not inject their code into every web page ever, unless the extension is actually used?

I guess we know why Grammarly never has any problems raising more funding.

39. MartijnHols ◴[] No.43516805[source]
Makes me wonder if you can use this to hijack their plugin. At the very least you should be able to inject text into it, but you can probably render a pretty little login form as well, abusing the trust the user has in their extension. Is injecting elements into a document controlled by others really safe?
replies(1): >>43517113 #
40. jgalt212 ◴[] No.43516812[source]
> At work we have a lot of sentry errors related to browser extensions doing weird stuff.

Are you referring to the "Object captured as exception" error which Sentry refuses to give any guidance on? We just end of filtering these out client-side.

replies(1): >>43517558 #
41. ◴[] No.43516846[source]
42. MartijnHols ◴[] No.43516871[source]
It's not just noise though; clients are actually experiencing crashes and other issues because of it. I wrote an in-depth article on the Google Translate extension's interference of React (and other webapps): https://martijnhols.nl/blog/everything-about-google-translat...

It's no wonder frontend has a lot more errors, after all it has to support so many more client variations than a typical backend. It can be very hard to make a big webapp that works well for everyone.

43. dbushell ◴[] No.43516920[source]
thanks!
replies(1): >>43517061 #
44. MartijnHols ◴[] No.43516925{3}[source]
Unfortunately browsers don't really provide good solutions for extensions that need to inject or change sites. Look at Google's owner in-browser translate extension, its DOM manipulation breaks many interactive apps as well. There are no tools available in browsers for it to not need to do that.
45. the8472 ◴[] No.43516957[source]
or better yet: use shadow dom
46. MartijnHols ◴[] No.43516979[source]
While Google Translate's interference sucks, with current browser tools, I don't think they can really operate any other way.

This is mostly because of cases where they need to translate a sentence like "[Click here] for more information". When translating it to another language, they may have to move the link to the end e.g. "For more information, [click here]". The only way to achieve that is to shuffle DOM elements around, which can cause interference with interactive apps.

There's still a lot the Google Translate team can do to reduce the interference they cause, but I don't think they can fully eliminate it without some new browser APIs.

replies(1): >>43518847 #
47. lelandfe ◴[] No.43516982[source]
I wonder what one variable could be injected to most break the web. I’m feeling:

  --primary-color: transparent
replies(1): >>43525267 #
48. mopenstein ◴[] No.43517008{3}[source]
I wonder what would happen if you cut your salary requirements in half. Would they still reject you on your age or what? And if so, would they reject you if you slashed it in half again? And keep slashing until they hired you. Just as an experiment
replies(2): >>43517149 #>>43517711 #
49. elros ◴[] No.43517018{3}[source]
In any case sounds like you dodged a bullet :-)
50. MasterScrat ◴[] No.43517036[source]
I'm out of the frontend field - what would be common ways to handle CSS namespacing these days?
replies(1): >>43517080 #
51. kstrauser ◴[] No.43517061{3}[source]
You bet!
52. echoangle ◴[] No.43517080{3}[source]
I don't know if this is what was meant by namespacing but i've seen '--{{ projectname}}-my-variable' before, so something like '--grammarly-rem' in this case.
53. echoangle ◴[] No.43517113[source]
How would this work? They are injecting CSS into your page, but you can't inject anything into the extension UI from a website. The only thing you could do would be to emulate the extension UI in your website, but for that you don't need to inject anything. You can just copy the design.
replies(1): >>43517231 #
54. mvid ◴[] No.43517149{4}[source]
“There is no ageism, because you could work for free or at a loss!”
replies(1): >>43517643 #
55. MartijnHols ◴[] No.43517231{3}[source]
The article mentions they inject a web component. I imagine a bad actor could add something to that. In this case at the very least the author could add a "I hacked your Grammarly extension" text just via CSS, but I'm sure you can go much further, even more so with other extensions (eg password managers).
replies(1): >>43517383 #
56. mrweasel ◴[] No.43517379[source]
I count myself fairly lucky to have colleague that care about these sorts of things. We have had meeting halted because it was obvious that some participants had certain extensions installed, AI assistants of various types, and some colleagues aren't comfortable with information potentially being picked up by a third party. So the meeting is halted until the extension is disabled.
57. echoangle ◴[] No.43517383{4}[source]
But you could also just add you own lookalike web component to you page that looks like the grammarly one. If people enter credentials there, it's user error.
58. emptysea ◴[] No.43517558{3}[source]
Ah yeah I remember that one, but can’t remember the origin.

A lot of times the reason sentry can’t do much is because the browser JS VMs have terrible/non-existent stack traces, especially true with things like unhandled rejected promises.

59. stavros ◴[] No.43517643{5}[source]
No but it's a good question. Is it "out of two candidates with similar experience and salary requirements, companies will usually hire the younger one", or is it "companies don't want expensive experienced people because cheaper, less-experienced people are good enough"?
replies(1): >>43528458 #
60. stavros ◴[] No.43517675[source]
This is fairly unrealted, but it irks me when one-line fixes like this sit for ages in backlog hell. I want a company where developers go "might as well fix that now, it's faster than writing a ticket for it".

I see people where I work not do this, and it drives me crazy. Our director of engineering will literally add tickets for himself to do things that would take less time to just do. At least I hear "I took a page from your book and messaged the person instead of adding a ticket for myself to message them" often, which is a good sign.

replies(2): >>43517770 #>>43517858 #
61. bryanrasmussen ◴[] No.43517711{4}[source]
I'm not somewhere you earn SV wages, so half my wage would be less than a junior gets paid in my country, and half that again would be less than I get for unemployment. Given market conditions I have dropped 10-15% off asking price but I don't think it is worth that much more for me to drop, given that if I can't get at that I probably can't get.

also, yeah, what the neighboring comment said - you do seem to think I should consider working for free...

62. quesera ◴[] No.43517770{3}[source]
> Our director of engineering will literally add tickets for himself to do things that would take less time to just do

I've done this. It irks me too, but sometimes I'm in organizational mode, and sometimes I'm in execution mode. :)

Also, I work in an environment[0] where all work is required to go through the formal tracker documentation flow (and all code changes must be approved by a second party).

So the ticket step is non-optional, and in fact required before work can begin -- we name branches with the ticket ID, so that Pivotal[1] can track the GitHub lifecycle.

[0] PCI-DSS, SOC 2, etc

[1] RIP :(

replies(1): >>43517789 #
63. stavros ◴[] No.43517789{4}[source]
Yeah, I guess if you're in a regulated environment, you have no choice. Most companies have no excuse, though!
64. kstrauser ◴[] No.43517858{3}[source]
To be clear, I'm answering this purely from a personal standpoint and not talking about any particular employer, and this doesn't relate to this specific issue at all.

Yes, I totally agree. The point of processes is to enable a company to manage huge amounts of potential work. Sometimes, processes can get in the way of just doing the simple thing we all know needs to be done.

Buuuut, I've been on the other side of that, too. Someone asks me to make some change. Sure! That's a reasonable idea and it would improve things. Making the change would take about 20 minutes. However, 437 different systems expect that thing to have its current behavior, and updating them to use the new behavior would be quite the project. In a vacuum, the change is simple and shouldn't take long to implement. Not many things operate in a vacuum, though.

For example, it would take like 5 minutes to do a "find all" in the Nginx source code and fix the misspelling of "referrer" as "referer". It would take a lot longer to update the standards with the correct spelling, and every client and other server to use it, would be slightly more challenging.

replies(1): >>43517914 #
65. stavros ◴[] No.43517914{4}[source]
Sure, but that's not what I'm talking about. That's not a change that takes 20 minutes, that's a change that takes years.

I'm talking about things like fixing a typo, where it literally takes multiple times the work to write the ticket than to grep for it, fix it, and push a PR.

replies(1): >>43517992 #
66. kstrauser ◴[] No.43517992{5}[source]
Yeah, I hear ya. Sometimes people seem to do bookkeeping just for the sheer joy of bookkeeping, or something.
67. ZeroTalent ◴[] No.43518609{4}[source]
> As if bots weren't enough of a problem, imagine when social media is just people clicking on buttons: "write a funny response", "write a comment in disagreement", "write 'same'", etc.

There are already extensions for this purpose.

replai.so and dozens others

68. olevzhyn ◴[] No.43518735[source]
Hey. I’m an engineer at Grammarly Extension. First of all, I’m really sorry that our extension broke the UX on dbushell.com and caused the author to spend time and effort figuring this out.

That was never intentional, and we are using various techniques to prevent this from happening. Unfortunately, that wasn’t enough. The article clearly shows that there’s room for improvement.

We temporarily added an exception for dbushell.com as a quick fix. In the meantime, we’re working on a change to ensure proper style isolation; such issues must never be the case.

Thank you!

replies(1): >>43520668 #
69. nikolay ◴[] No.43518739[source]
I am happy with Microsoft's free grammar checker extension - Microsoft Editor [0], which supports foreign languages as well... although I still pay for Grammarly. Microsoft's works more smoothly and on more sites, including Hacker New!

[0]: https://chromewebstore.google.com/detail/microsoft-editor-sp...

70. kelvinjps10 ◴[] No.43518847{3}[source]
But they are the owners of a browser
71. kelvinjps10 ◴[] No.43518857{3}[source]
But wouldn't you brake their extensions if the user wanted them to work?
72. kelvinjps10 ◴[] No.43518863{3}[source]
Not even ublock?
73. b0ner_t0ner ◴[] No.43520668[source]
Is Grammarly hiring?
replies(1): >>43522285 #
74. kstrauser ◴[] No.43522285{3}[source]
We are! https://www.grammarly.com/careers/jobs
75. xigoi ◴[] No.43525267[source]

  --serif: "Comic Sans MS"
76. mopenstein ◴[] No.43528458{6}[source]
Thank you. Others took it so personally.

Your observation is spot on.

replies(1): >>43528874 #
77. stavros ◴[] No.43528874{7}[source]
No problem, I have the same question. I'd like to know if it's "old people can't be good at the job" or "we can't afford experienced people". In my experience, it's more the latter.
replies(1): >>43530864 #
78. therealdrag0 ◴[] No.43530864{8}[source]
Old people can be good. But young people are more likely to be hungry/ambitious and take ownership, which (at least has the appearance of) produces more/better outcomes.