DEV Community

Tom Hughes
Tom Hughes

Posted on • Edited on

Building My First Trading Bot (And How It Lost All My Money)

It was 2017 and I’d just moved to Thailand. I was loving every minute of it; Working from coffee shops, going to events at night and mingling with other people who were also working online, or at least attempting to.

Life was good.

The only issue was that I was hemorrhaging money, it was a small hemorrhage (cost of living is low in Thailand), but a hemorrhage nonetheless.

I had no income source. My justification for this was that I was learning to write code, and soon enough this new knowledge would be able to provide me with all the money I’d ever need.

I loved the idea of being able to make anything with JUST a laptop and an internet connection, I was determined to make this lifestyle work.


The Dream

I had a great idea. I was going to build a program that would automatically place bets for me.

Furthermore, this bot would be intelligent, it would know exactly how much money I had and would increase / decrease the amount bet accordingly.

If this worked, I could sit back and drink yai leos (large beers) while my bot diligently printed money.

The idea was simple, I’d subscribe to lots of different betting ‘tipsters’, and any selections that more than one tipster recommended, I’d bet on it.

These tipsters were all claiming to be profitable gamblers, so my thinking was that by only betting on the ones that more than one tipster recommended, I’d be increasing the profitability of the bets.

Building the Dream

I discovered a pretty clever way to input all the data. Whenever a tipster would send me an email with the days recommended tips, I would forward it to my bot. The bot would then parse out the information and put it into a big database.

An example email that would be parsed into JSON

Once I had all the day's selections in the database, I needed a way to be able to tell if there were any duplicate tips.

This is actually harder than it sounds. There was a range of inconsistencies between the way each tipster sent out their tips, such as;

Kempsey 18:30 #5 Little Miss Sunshine
Kempsey 6:30pm #5 Little Ms Sunshine

Notice the inconsistent use of apostrophes, 24hr time, abbreviations etc.

All this made it very hard to actually match the data together. This led me into the topic of ‘fuzzy matching’.

Fuzzy Matching attempts to match datasets that aren’t 100% the same.

The first thing I did was attempt to normalize all the data, by stripping out all the apostrophes and special characters, and converting everything to lowercase.

Then I found a library that could detect and convert 12 hour time into 24 hour time, so I used that.

The beauty about programming is that you don’t need to do everything yourself, there’s public online libraries where you can pull and use other peoples (often much more sophisticated) code. Perfect for a newbie like me!

Now the above example looked like this;

kempsey 18:30 #5 little miss sunshine
kempsey 18:30 #5 little ms sunshine

Finally I found an algorithm known as the Levenshtein Algorithm. This is a form of fuzzy matching that compares how many substitutions are required to make two strings the same.

For example ‘dog’ compared to ‘dag’ has a Levenshtein distance of 1, because all we need to do to make the words match is replace ‘a’ with ‘o’.

I decided that any two tips that had a Levenshtein distance lower than 5 would be assumed to be the same.

Input 1 Input 2 Levenshtein Distance The Same?
kempsey 18:30 #5 little miss sunshine kempsey 18:30 #5 little ms sunshine 2 Yes
kempsey 18:30 #5 little miss sunshine kempsey 15:00 #3 dance monkey 21 No

Now that all the hard stuff was out of the way, all I had to do was implement the actual placing of the bets.

There’s a lot more technical stuff involved in how this was made, but parsing and normalizing the tips was by far the most challenging part of the build.

All my bets were going to be placed on Betfair, which is a betting exchange where people can place bets against other punters. Betfair is great because they have a really easy to use API.

Using the Betfair API to place the bets was pretty straight forward. I ran a query to get a list of all of the horses that were running today, then found the ID of the horse that corresponded to the horse I wanted to bet on (also using the Levenshtein algorithm) and I was off to the races (literally).

Now, it’s safe to say that this whole codebase was a sloppy mess. I’m no zealot about writing clean efficient code, but this was sloppy even by my standards (to be fair, I was still learning).

Sloppiness aside, I was done.

I managed to build a bot that would receive emails, parse data, normalize that data and then query Betfair’s API to place the bets using real money.

100% automated, no input required by me, brilliant!

It would even send me an email with the bets it was placing (as shown below).

An example email the bot would send me with the days bets

The Dream Gone Wrong

Now all there was to do was fund my Betfair account and let the bot work it’s magic!

As I was still a complete beginner, I decided to only bet a TINY amount of money.

The amount I was going to bet would be determined by the odds that the horse was suggested at. I would always bet to profit a certain amount of money, around $100.

Odds Stake Return Profit
10 $11 $110 $100
5 $25 $125 $100
2.5 $66.6 $166.6 $100

Notice how the lower the odds, the more I bet?

This is good because I’d be betting more money on the more horses that were more likely to win, and less on the outsiders.

After a few days of the bot running I received this email.

An Email the bot sent with very strange data

Oh no.

I knew something was horribly wrong because I would never bet on a horse at odds of 1.08.

Because of the way I was staking, I knew that this was likely going to result in my bot betting my ENTIRE BALANCE on this one horse.

Suddenly the nice message I programmed the bot to send “have a great night” was like a kick in the nuts.

“Hey Tom, I’ve bet all your money. Have a great night”

Sure enough, I log into my Betfair account and see the following.

Betslip showing $2304 bet

Ouch, instead of betting a tiny amount of money like I wanted, it bet over $2000.

Considering I didn’t have an income source, this was a devastating blow.

What the hell went wrong?

The Aftermath

I went back into my emails and found the issue.

One of the tipsters emails that I was parsing had added in a new column unexpectedly.

The column that my bot normally would find the suggested odds, was actually a new column called ‘value for money’, a number between 1 and 1.2.

The ACTUAL suggested odds for this horse was 6.5.

But not all hope was lost. Betfair is a market where you can buy AND sell bets.

Maybe I could just sell the bet back to the market? Not quite it turns out.

Betfair operates like an exchange. People put in their orders that they are willing to buy and sell tips at. Because my bot put $2304 on this horse at odds of 1.08, Betfair matched my bet with ANY order that was higher than 1.08.

Considering the market odds for this horse was actually 6.5, I accepted pretty much every order that was available. No price was too low for me! I chowed through that order book like Pacman.

Pacman gif

This can sometimes happen in the stock market, and is known as a ‘fat thumb’ (lol).

Literally because someone with a ‘fat thumb’ accidentally hits the wrong key on the keyboard (because of their enormous thumb) and buys/sells WAY more than they want to.

This has the result of eating through the order book, much like my bet did. Below is an example of what this looks like;

An example of a fat thumb error

Notice the big long red line? That’s a textbook fat thumb. Someone with a lot of money accidentally sold $10 million of something instead of $10,000.

After a ‘fat thumb’ event, the market quickly re-stabilizes as seen above. It’s nothing more than a blip.

Because of this phenomenon, the odds for the horse that I bet $2304 on quickly rose back up to the market average of 6.5.

Given that I got average odds of 2.56, there was no way I could sell the bet without guaranteeing a massive loss. I decided to ride the bet.

I thought about writing a big paragraph about how nervous and excited I was to watch the race, imagining the glory if it won, but it’s not really the point of this post. Long story short, the horse lost and with it went my $2304.

I deactivated the bot pretty quickly after this.

The Postmortem

Looking back now as an experienced developer, there’s literally infinite things I could have done to prevent this.

The most obvious (and simple) one is implementing what’s known as a guard clause.

A guard clause is a boolean expression that HAS to evaluate to true in order for a function to continue executing.

As an example, a function that fails if any of its inputs are negative might have a guard clause like;

return if input < 0
Enter fullscreen mode Exit fullscreen mode

It’s a simple check that everything is as it should be before running.

Below is the function I wrote that is responsible for placing the bet (written in ruby);

def place_bet(market, horse, stake, odds)
    @order = @client.place_orders({
        marketId: market,
        instructions: [{
            orderType: "LIMIT",
            selectionId: horse,
            side: "BACK",
            limitOrder: {
                size: stake,
                price: odds,
                persistenceType: "MARKET_ON_CLOSE"
            }
        }]
    })
    @order
end
Enter fullscreen mode Exit fullscreen mode

Notice there’s no checks or anything. This could place a bet for $1,000,000 and my bot wouldn’t bat an eyelid.

I could add one line of code to fix that;

return if stake > 200
Enter fullscreen mode Exit fullscreen mode

That’s it. Now any bet > $200 would be considered invalid and wouldn't be placed.

That simple, obvious line of code could have saved me $2304.

That is, until the next bug comes along.

Thanks for reading, if you enjoyed this post please subscribe to my blog (https://tomhughes.cc). I write about my experiences as a solo developer, what it’s like living overseas, gambling/investing as an income and more :)

Top comments (2)

Collapse
 
bradnichol profile image
Bradley Nichol

Great read that Tom, thanks.
What a difference a single line of code can make 😆

Collapse
 
thughes24 profile image
Tom Hughes

Thanks Bradley, glad you liked it :)