Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to create and use nonces

I am running a website, and there is a scoring system that gives you points for the number of times you play a game.

It uses hashing to prove the integrity of http request for scoring so users cannot change anything, however as I feared might happen, someone figured out that they didn't need to change it, they just needed to get a high score, and duplicate the http request, headers and all.

Previously I'd been prohibited from protecting against this attack because it was considered unlikely. However, now that it has happened, I can. The http request originates from a flash game, and then is validated by php and php enters it into the database.

I'm pretty sure nonces will solve the issue, but I'm not exactly sure how to implement them. What is a common, and secure way of setting up a nonce system?

like image 320
Malfist Avatar asked Nov 10 '10 14:11

Malfist


People also ask

How are nonces created?

A random nonce is produced by stringing arbitrary numbers together. A sequential nonce is produced incrementally. Using the sequential nonce method guarantees that values are not repeated, cannot be replayed and do not take up unnecessary space.

How is a nonce used?

In cryptocurrency, a nonce is an abbreviation for "number only used once," which is a number added to a hashed—or encrypted—block in a blockchain that, when rehashed, meets the difficulty level restrictions. The nonce is the number that blockchain miners are solving to receive the block reward.

What is a nonce example?

“Jabberwocky:" “Jabberwocky” (a nonce word itself) is a famous Lewis Carroll nonsense poem that appears in his novel, Through the Looking Glass (1871). The poem contains plenty of nonce words, such as “brillig,” which in the poem means “four in the afternoon,” but does not have an official meaning anywhere else.


1 Answers

It's actually quite easy to do... There are some libraries out there to do it for you:

  1. PHP Nonce Library
  2. OpenID Nonce Library

Or if you want to write your own, it's pretty simple. Using the WikiPedia page as a jumping off point, In pseudo-code:

On the server side, you need two client callable functions

getNonce() {     $id = Identify Request //(either by username, session, or something)     $nonce = hash('sha512', makeRandomString());     storeNonce($id, $nonce);     return $nonce to client; }  verifyNonce($data, $cnonce, $hash) {     $id = Identify Request     $nonce = getNonce($id);  // Fetch the nonce from the last request     removeNonce($id, $nonce); //Remove the nonce from being used again!     $testHash = hash('sha512',$nonce . $cnonce . $data);     return $testHash == $hash; } 

And on the client side:

sendData($data) {     $nonce = getNonceFromServer();     $cnonce = hash('sha512', makeRandomString());     $hash = hash('sha512', $nonce . $cnonce . $data);     $args = array('data' => $data, 'cnonce' => $cnonce, 'hash' => $hash);     sendDataToClient($args); } 

The function makeRandomString really just needs to return a random number or string. The better the randomness, the better the security... Also note that since it's fed right into a hash function, the implementation details don't matter from request to request. The client's version and the server's version don't need to match. In fact, the only bit that needs to match 100% is the hash function used in hash('sha512', $nonce . $cnonce . $data);... Here's an example of a reasonably secure makeRandomString function...

function makeRandomString($bits = 256) {     $bytes = ceil($bits / 8);     $return = '';     for ($i = 0; $i < $bytes; $i++) {         $return .= chr(mt_rand(0, 255));     }     return $return; } 
like image 91
ircmaxell Avatar answered Oct 31 '22 17:10

ircmaxell