Technology

Emulate User Activity with Bots

#Emulate User Activity with Bots

In our sales events platform, Auctibles, sellers have the exciting opportunity to create various events. These range from a sale event at a fixed price to live auctions, where buyers bid the price up, providing a dynamic and engaging experience.

#The Technology

We use the Laravel framework in the context of the TALL stack: Tailwind, Alpine.js, Laravel, and Livewire.

#Why Emulation?

For a demo environment, it's crucial that we accurately emulate buyers' activity for a seller. This includes sending buy orders and chat messages. The purpose of the emulation is to allow the seller to get a feeling of how the event with multiple buyers will be observed from the event control screens. This includes changing the quantities of items still available, dynamic price changes, and chat messages from buyers.

#Ordinary Buyers

The buy screen is a Livewire component. Clicking the BUY button sends a broadcast event: LiveBuy, LiveBid,...

Similarly, buyers send chat messages via a broadcast event ChatMessage.

#Bots for Emulation

We emulate buyers' activity via a bot. The bot is a ReactPHP server that periodically sends the same broadcast events that ordinary buyers send.

The server receives a set_timer GET call at the event's start and a stop_timer GET call at its end.

$loop = Factory::create();
$server = new Server(function (ServerRequestInterface $request) use($loop) {
    
        $path = $request->getUri()->getPath();
        $method = $request->getMethod();
        
        if ($path == '/set_timer' && $method === 'GET') {

          $params = $request->getQueryParams();
		        $event_id = $params['event_id'] ?? null;
		        
		        // start emulation
		        
		        return Response::plaintext('timer set')->withStatus(Response::STATUS_OK);   
		        
	        } else if ($path == '/stop_timer' && $method === 'GET') {

            $params = $request->getQueryParams();
	            $event_id = $params['event_id'] ?? null;   
	            
	            // stop emulation
	            
	            return Response::plaintext('timer stopped')->withStatus(Response::STATUS_OK);               
					}
    
});
    
$socket = new SocketServer('0.0.0.0:' . config('app.BOT_SERVER_PORT'));
$server->listen($socket);

$loop->run();

#Laravel to ReactPHP

These calls are received from the component where the seller controls the event, using Http:

  $url = config('app.BOT_SERVER_URL') . ':' . config('app.BOT_SERVER_PORT') . '/set_timer';
  $data = [
  'event_id' => $this->event->id,
  ];

$response = Http::withQueryParameters($data)->get($url);

#Starting the Bot

To start the emulation, we set a periodic timer,

$timer = $loop->addPeriodicTimer($this->period_in_seconds, function () use ($event_id) {
                           
   $this->send_bid($event_id);
   $this->send_message($event_id);

});

We store the timer in an array:

$this->timers[$event_id] = $timer;

#Stopping the Bot

$loop->cancelTimer($this->timers[$event_id]);
unset($this->timers[$event_id]);
Yoram Kornatzky

Yoram Kornatzky