siunam's Website

My personal website

Home Writeups Research Blog Projects About

terms-and-conditions

Table of Contents

  1. Overview
  2. Background
  3. Enumeration
  4. Get the Flag
  5. Conclusion

Overview

Background

Welcome to LA CTF 2024! All you have to do is accept the terms and conditions and you get a flag!

terms-and-conditions.chall.lac.tf

Enumeration

Index page:

In here, we can see that there's a button called "I Accept".

However, when my cursor gets closer to the button, the button goes away from my cursor:

Hmm… Why?

Well, if you view the source by using key Ctrl + U, you can see some JavaScript code:

            [...]
            <button tabindex="-1" id="accept">I Accept</button>
        [...]
        <script defer src="/analytics.js"></script>
        [...]
        <script id="mainscript">
            const accept = document.getElementById("accept");
            document.body.addEventListener("touchstart", (e) => {
                document.body.innerHTML = "<div><h1>NO TOUCHING ALLOWED</h1></div>";
            });
            let tx = 0;
            let ty = 0;
            let mx = 0;
            let my = 0;
            window.addEventListener("mousemove", function (e) {
                mx = e.clientX;
                my = e.clientY;
            });
            setInterval(function () {
                const rect = accept.getBoundingClientRect();
                const cx = rect.x + rect.width / 2;
                const cy = rect.y + rect.height / 2;
                const dx = mx - cx;
                const dy = my - cy;
                const d = Math.hypot(dx, dy);
                const mind = Math.max(rect.width, rect.height) + 10;
                const safe = Math.max(rect.width, rect.height) + 25;
                if (d < mind) {
                    const diff = mind - d;
                    if (d == 0) {
                        tx -= diff;
                    } else {
                        tx -= (dx / d) * diff;
                        ty -= (dy / d) * diff;
                    }
                } else if (d > safe) {
                    const v = 2;
                    const offset = Math.hypot(tx, ty);
                    const factor = Math.min(v / offset, 1);
                    if (offset > 0) {
                        tx -= tx * factor;
                        ty -= ty * factor;
                    }
                }
                accept.style.transform = `translate(${tx}px, ${ty}px)`;
            }, 1);
            let width = window.innerWidth;
            let height = window.innerHeight;
            setInterval(function() {
                if (window.innerHeight !== height || window.innerWidth !== width) {
                    document.body.innerHTML = "<div><h1>NO CONSOLE ALLOWED</h1></div>";
                    height = window.innerHeight;
                    width = window.innerWidth;
                }
            }, 10);
        </script>
        [...]

More specifically, take a look at the following JavaScript code:

const accept = document.getElementById("accept");
document.body.addEventListener("touchstart", (e) => {
    document.body.innerHTML = "<div><h1>NO TOUCHING ALLOWED</h1></div>";
});

First, get the <button> element with id accept by using DOM (Document Object Model) method getElementById(). Then, the JavaScript adds an event called touchstart to that button. When there's a touchstart event is being fired, the HTML body text change to <div><h1>NO TOUCHING ALLOWED</h1></div>.

According to https://www.w3schools.com/jsref/event_touchstart.asp, the touchstart event only works on touch screens. So, if we're using touch screen to touch the accept button, the body text will be changed to the "NO TOUCHING ALLOWED".

Next, we can see why the accept button is moving away from our cursor:

let tx = 0;
let ty = 0;
let mx = 0;
let my = 0;
window.addEventListener("mousemove", function (e) {
    mx = e.clientX;
    my = e.clientY;
});
setInterval(function () {
    const rect = accept.getBoundingClientRect();
    const cx = rect.x + rect.width / 2;
    const cy = rect.y + rect.height / 2;
    const dx = mx - cx;
    const dy = my - cy;
    const d = Math.hypot(dx, dy);
    const mind = Math.max(rect.width, rect.height) + 10;
    const safe = Math.max(rect.width, rect.height) + 25;
    if (d < mind) {
        const diff = mind - d;
        if (d == 0) {
            tx -= diff;
        } else {
            tx -= (dx / d) * diff;
            ty -= (dy / d) * diff;
        }
    } else if (d > safe) {
        const v = 2;
        const offset = Math.hypot(tx, ty);
        const factor = Math.min(v / offset, 1);
        if (offset > 0) {
            tx -= tx * factor;
            ty -= ty * factor;
        }
    }
    accept.style.transform = `translate(${tx}px, ${ty}px)`;
}, 1);

In here, the event mousemove has been added to the accept button. As the event name says, it's fired when the mouse is moving. In our case, when we move the mouse/cursor, it tracks our cursor's X (e.clientX) and Y (e.clientY) position.

Then, by using global function setInterval(), every 1 millisecond it'll check our cursor position is within the safe zone of the accept button's position. If not, it uses CSS style to change the position of the accept button (translate()).

Hmm… How can we click the accept button without touching/getting near it??

If you take a closer look at the last pieces of the index page's JavaScript code, you'll find something called "browser console":

let width = window.innerWidth;
let height = window.innerHeight;
setInterval(function() {
    if (window.innerHeight !== height || window.innerWidth !== width) {
        document.body.innerHTML = "<div><h1>NO CONSOLE ALLOWED</h1></div>";
        height = window.innerHeight;
        width = window.innerWidth;
    }
}, 10);

In here, for every 10 milliseconds, if our current window's height and width is not the same as the newly retrieved one, our body HTML will become <div><h1>NO CONSOLE ALLOWED</h1></div>.

However, when we refresh the page, not only the browser console is still there, but also the check will not get passed.

When we refresh the page, our width and height is the window size with the browser console, so when the JavaScript retrieve our window size again, the width and height is the same as our current window size.

Get the Flag

Now, in Firefox, you can open the browser console via hitting F12:

As expected, our current window size has been changed when the browser console appeared.

However, when we refresh the page, our body HTML won't change to NO CONSOLE ALLOWED:

Now, we can click the accept button with JavaScript code!

But before we do that, let's try to understand what will happen when we click that button.

During view the index page's source page, there's another JavaScript file has been loaded:

<script defer src="/analytics.js"></script>

Oh boy… It's obfuscated… And I don't want to deobfuscate it…

Obfuscation is a technique that turns the code into a very unreadable way.

Anway, let's just click the accept button.

In JavaScript's DOM, there's a method called click(), which allows us to click an element.

Since the accept button has already declared (See the 1 line above the touchstart event), there's no need to declare it again or get the element with DOM method getElementById().

Hence, we can click the accept button by typing the following JavaScript code in the browser console:

accept.click()

Here we go! There's the flag!

Conclusion

What we've learned:

  1. Debugging via browser console