Contents

๐ŸŒ WebSockFish

A detailed write-up of the Web challenge 'WebSockFish' from PicoCTF - 2025

/images/PicoGym/PicoCTF-2025/WebSockFish/challenge_presentation.png
Challenge Presentation

๐Ÿ“Š Challenge Overview

Category Details Additional Info
๐Ÿ† Event PicoGym Event Link
๐Ÿ”ฐ Category Web ๐ŸŒ
๐Ÿ’Ž Points Out of 500 total
โญ Difficulty ๐ŸŸก Medium Personal Rating: 2/10
๐Ÿ‘ค Author Venax Profile
๐ŸŽฎ Solves (At the time of flag submission) 1.321 solve rate
๐Ÿ“… Date 20-03-2025 PicoGym
๐Ÿฆพ Solved By mH4ck3r0n3 Team:

๐Ÿ“ Challenge Information

Can you win in a convincing manner against this chess bot? He won’t go easy on you!

๐ŸŽฏ Challenge Files & Infrastructure

Provided Files

1
Files: None

๐Ÿ” Initial Analysis

First Steps

Initially, the website appears as follows:

/images/PicoGym/PicoCTF-2025/WebSockFish/site_presentation.png
Site Presentation

the first thing I thought of was using stockfish to beat stockfish, but after winning the game, I didn’t get any flag. So I started looking for other ways to win the game. Inspecting the source of the page, I noticed that everything is managed client-side, and websockets are used to communicate with js/stockfish.min.js:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
var ws_address = "ws://" + location.hostname + ":" + location.port + "/ws/";  
const ws = new WebSocket(ws_address);  
ws.onmessage = (event) => {  
   const message = event.data;  
   updateChat(message);  
};  
function sendMessage(message) {  
   ws.send(message);  
}  
function updateChat(message) {  
   const chatText = $("#chatText");  
   chatText.text(message);  
}  

so far, the most interesting function is updateChat(message), since it is the function that I assume changes the stockfish message, as seen in the previous screenshot. Intuitively, I think thatโ€™s where we somehow need to retrieve the flag. Continuing to read, I found the function that stockfish uses to handle messages:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
stockfish.onmessage = function (event) {  
   var message;  
   // console.log(event.data);  
   if (event.data.startsWith("bestmove")) {  
       var bestMove = event.data.split(" ")[1];  
       var srcSq = bestMove.slice(0, 2);  
       var dstSq = bestMove.slice(2, 4);  
       var promotion = bestMove.slice(4);  
       game.move({ from: srcSq, to: dstSq, promotion: promotion });  
       board.position(game.fen());  
   } else if (event.data.startsWith(`info depth ${DEPTH}`)) {  
       var splitString = event.data.split(" ");  
       if (event.data.includes("mate")) {  
           message = "mate " + parseInt(splitString[9]);  
       } else {  
           message = "eval " + parseInt(splitString[9]);  
       }  
       sendMessage(message);  
   }  
};  

since we can send messages using the ws.send("message") function (as written in the sendMessage() function seen earlier), I started testing. In fact, by sending bestmove as a message, the stockfish text shouldn’t change based on the function just seen:

/images/PicoGym/PicoCTF-2025/WebSockFish/bestmove.png
Best Move

as we can see, the sendMessage(message) function is not called, and the message remains unchanged. Now let’s try with mate:

/images/PicoGym/PicoCTF-2025/WebSockFish/mate.png
Mate

as we can see, stockfish interprets the message mate 0 as checkmate in zero moves. However, in a situation where checkmate does not occur, eval x is sent. Let’s see what this eval does:

/images/PicoGym/PicoCTF-2025/WebSockFish/eval.png
Eval

by sending eval 0, I get the message I think this position is pretty equal. From here, it can be inferred that eval x changes the stockfish message as we move pieces, analyzing the game, and the message will be adjusted based on how the game is going. In case of a disadvantage, it will say something related to it, and similarly in case of an advantage. Let’s move on to the exploitation phase.

๐ŸŽฏ Solution Path

Exploitation Steps

Initial setup

As mentioned earlier, the flag will most likely be contained in the messages from stockfish, so I wondered, since sending 0 results in an equal position, what would happen if I sent 9999?

/images/PicoGym/PicoCTF-2025/WebSockFish/eval9999.png
Eval 9999

Given the message You're in deep water now!, I imagine that with positive numbers, it seems stockfish believes it has an advantage, and consequently, I think that with negative numbers, it would think it is at a disadvantage.

Exploitation

So I initially tried with low negative numbers since I didnโ€™t know how far stockfishโ€™s scoring goes. Later, when I sent -99999, stockfish responded with the flag message, thinking it was at such a big disadvantage that it forced it to surrender.

Flag capture

/images/PicoGym/PicoCTF-2025/WebSockFish/manual_flag.png
Manual Flag

๐Ÿ› ๏ธ Exploitation Process

Approach

The automatic exploit connects to a websocket, sends the message eval -99999, captures the response, and extracts the flag from it using a regex.

๐Ÿšฉ Flag Capture

Flag

picoCTF{c1i3nt_s1d3_w3b_s0ck3t5_e5e75e69}

Proof of Execution

/images/PicoGym/PicoCTF-2025/WebSockFish/automated_flag.png
Automated Flag
Screenshot of successful exploitation

๐Ÿ”ง Tools Used

Tool Purpose
Python Exploit
WebDevTools Web Testing

๐Ÿ’ก Key Learnings

Time Optimization

  • Try to understand how things work with some testing.

Skills Improved

  • Binary Exploitation
  • Reverse Engineering
  • Web Exploitation
  • Cryptography
  • Forensics
  • OSINT
  • Miscellaneous

๐Ÿ“š References & Resources

Learning Resources


๐Ÿ“Š Final Statistics

Metric Value Notes
Time to Solve 00:10 From start to flag
Global Ranking (At the time of flag submission) Challenge ranking
Points Earned Team contribution

Created: 20-03-2025 โ€ข Last Modified: 20-03-2025 *Author: mH4ck3r0n3 โ€ข Team: *