2015-01-12 21:51:52 +00:00
#!/bin/python
#
# Cryptonote tipbot - dice commands
# Copyright 2014,2015 moneromooo
#
# The Cryptonote tipbot is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License as published
# by the Free Software Foundation; either version 2, or (at your option)
# any later version.
#
import sys
import os
import redis
import hashlib
import time
import string
import random
import tipbot . config as config
from tipbot . log import log_error , log_warn , log_info , log_log
import tipbot . coinspecs as coinspecs
2015-02-14 12:15:49 +00:00
from tipbot . user import User
from tipbot . link import Link
2015-01-12 21:51:52 +00:00
from tipbot . utils import *
from tipbot . command_manager import *
from tipbot . redisdb import *
from tipbot . betutils import *
2015-01-13 12:28:05 +00:00
def Roll ( link ) :
identity = link . identity ( )
2015-01-12 21:51:52 +00:00
try :
2015-01-13 12:28:05 +00:00
if redis_hexists ( ' dice:rolls ' , identity ) :
rolls = redis_hget ( ' dice:rolls ' , identity )
2015-01-12 21:51:52 +00:00
rolls = long ( rolls ) + 1
else :
rolls = 1
except Exception , e :
2015-01-13 12:28:05 +00:00
log_error ( ' Failed to prepare roll for %s : %s ' % ( identity , str ( e ) ) )
2015-01-12 21:51:52 +00:00
raise
try :
2015-01-13 12:28:05 +00:00
s = GetServerSeed ( link , ' dice ' ) + " : " + GetPlayerSeed ( link , ' dice ' ) + " : " + str ( rolls )
2015-01-12 21:51:52 +00:00
sh = hashlib . sha256 ( s ) . hexdigest ( )
2015-01-31 13:01:03 +00:00
roll = float ( long ( sh [ 0 : 8 ] , base = 16 ) ) / 0xffffffff
2015-01-12 21:51:52 +00:00
return rolls , roll
except Exception , e :
2015-01-13 12:28:05 +00:00
log_error ( ' Failed to roll for %s : %s ' % ( identity , str ( e ) ) )
2015-01-12 21:51:52 +00:00
raise
2015-01-13 12:28:05 +00:00
def Dice ( link , cmd ) :
identity = link . identity ( )
2015-01-12 21:51:52 +00:00
try :
amount = float ( cmd [ 1 ] )
2015-02-02 12:22:20 +00:00
units = StringToUnits ( cmd [ 1 ] )
2015-01-12 21:51:52 +00:00
multiplier = float ( cmd [ 2 ] )
2015-01-17 09:02:37 +00:00
overunder = GetParam ( cmd , 3 )
2015-01-12 21:51:52 +00:00
except Exception , e :
2015-01-13 12:28:05 +00:00
link . send ( " Usage: dice amount multiplier [over|under] " )
2015-01-12 21:51:52 +00:00
return
2015-01-25 12:36:04 +00:00
if multiplier < float ( config . dice_min_multiplier ) or multiplier > float ( config . dice_max_multiplier ) :
link . send ( " Invalid multiplier: should be between %f and %f " % ( config . dice_min_multiplier , config . dice_max_multiplier ) )
2015-01-12 21:51:52 +00:00
return
2015-01-17 09:02:37 +00:00
if overunder == " over " :
under = False
elif overunder == " under " or not overunder :
under = True
else :
2015-01-13 12:28:05 +00:00
link . send ( " Usage: dice amount multiplier [over|under] " )
2015-01-17 09:02:37 +00:00
return
2015-01-12 21:51:52 +00:00
2015-01-13 12:28:05 +00:00
log_info ( " Dice: %s wants to bet %s at x %f , %s target " % ( identity , AmountToString ( units ) , multiplier , " under " if under else " over " ) )
2015-01-12 21:51:52 +00:00
potential_loss = amount * multiplier
2015-01-21 18:15:49 +00:00
valid , reason = IsBetValid ( link , amount , config . dice_min_bet , config . dice_max_bet , potential_loss , config . dice_max_loss , config . dice_max_loss_ratio )
2015-01-12 21:51:52 +00:00
if not valid :
2015-01-13 12:28:05 +00:00
log_info ( " Dice: %s ' s bet refused: %s " % ( identity , reason ) )
link . send ( " %s : %s " % ( link . user . nick , reason ) )
2015-01-12 21:51:52 +00:00
return
try :
2015-01-13 12:28:05 +00:00
rolls , roll = Roll ( link )
2015-01-12 21:51:52 +00:00
except :
2015-01-13 12:28:05 +00:00
link . send ( " An error occured " )
2015-01-12 21:51:52 +00:00
return
2015-01-19 19:39:43 +00:00
target = ( 1 - config . dice_edge ) / multiplier
2015-01-18 09:05:59 +00:00
if not under :
target = 1 - target
2015-01-13 12:28:05 +00:00
log_info ( " Dice: %s ' s # %d roll: %.16g , target %s %.16g " % ( identity , rolls , roll , " under " if under else " over " , target ) )
2015-01-12 21:51:52 +00:00
lose_units = units
2015-01-19 19:39:43 +00:00
win_units = long ( units * multiplier ) - lose_units
2015-01-12 21:51:52 +00:00
log_log ( ' units %s , multiplier %f , edge %f , lose_units %s , win_units %s ' % ( AmountToString ( units ) , multiplier , config . dice_edge , AmountToString ( lose_units ) , AmountToString ( win_units ) ) )
2015-01-17 09:02:37 +00:00
if under :
win = roll < = target
else :
win = roll > = target
2015-01-12 21:51:52 +00:00
if win :
2015-01-19 19:39:43 +00:00
msg = " %s bets %s and wins %s on roll # %d ! %.16g %s %.16g " % ( link . user . nick , AmountToString ( lose_units ) , AmountToString ( win_units + lose_units ) , rolls , roll , " <= " if under else " >= " , target )
2015-01-12 21:51:52 +00:00
else :
2015-01-19 19:39:43 +00:00
msg = " %s bets %s and loses on roll # %d . %.16g %s %.16g " % ( link . user . nick , AmountToString ( lose_units ) , rolls , roll , " > " if under else " < " , target )
2015-01-12 21:51:52 +00:00
try :
2015-01-13 12:28:05 +00:00
RecordGameResult ( link , " dice " , win , not win , win_units if win else lose_units )
2015-01-12 21:51:52 +00:00
except :
return
2015-01-13 12:28:05 +00:00
redis_hset ( " dice:rolls " , identity , rolls )
2015-01-12 21:51:52 +00:00
2015-01-13 12:28:05 +00:00
link . send ( " %s " % msg )
2015-01-12 21:51:52 +00:00
2015-01-13 12:28:05 +00:00
def ShowDiceStats ( link , sidentity , title ) :
return ShowGameStats ( link , sidentity , title , " dice " )
2015-01-12 21:51:52 +00:00
2015-01-13 12:28:05 +00:00
def GetDiceStats ( link , cmd ) :
identity = link . identity ( )
sidentity = GetParam ( cmd , 1 )
if sidentity :
sidentity = IdentityFromString ( link , sidentity )
if sidentity and sidentity != identity :
if not IsAdmin ( link ) :
log_error ( ' %s is not admin, cannot see dice stats for %s ' % ( identity , sidentity ) )
link . send ( ' Access denied ' )
2015-01-12 21:51:52 +00:00
return
else :
2015-01-13 12:28:05 +00:00
sidentity = identity
2015-01-21 22:00:22 +00:00
ShowDiceStats ( link , sidentity , NickFromIdentity ( sidentity ) )
ShowDiceStats ( link , " reset: " + sidentity , ' %s since reset ' % NickFromIdentity ( sidentity ) )
2015-01-13 12:28:05 +00:00
ShowDiceStats ( link , ' ' , ' overall ' )
2015-01-12 21:51:52 +00:00
2015-01-13 12:28:05 +00:00
def ResetDiceStats ( link , cmd ) :
identity = link . identity ( )
sidentity = GetParam ( cmd , 1 )
if sidentity :
sidentity = IdentityFromString ( link , sidentity )
if sidentity and sidentity != identity :
if not IsAdmin ( link ) :
log_error ( ' %s is not admin, cannot see dice stats for %s ' % ( identity , sidentity ) )
link . send ( ' Access denied ' )
2015-01-12 21:51:52 +00:00
return
else :
2015-01-13 12:28:05 +00:00
sidentity = identity
2015-01-12 21:51:52 +00:00
try :
2015-01-13 12:28:05 +00:00
ResetGameStats ( link , sidentity , " dice " )
2015-01-12 21:51:52 +00:00
except Exception , e :
2015-01-13 12:28:05 +00:00
link . send ( " An error occured " )
2015-01-12 21:51:52 +00:00
2015-01-13 12:28:05 +00:00
def PlayerSeed ( link , cmd ) :
identity = link . identity ( )
2015-01-12 21:51:52 +00:00
fair_string = GetParam ( cmd , 1 )
if not fair_string :
2015-01-13 12:28:05 +00:00
link . send ( " Usage: !playerseed <string> " )
2015-01-12 21:51:52 +00:00
return
try :
2015-01-13 12:28:05 +00:00
SetPlayerSeed ( link , ' dice ' , fair_string )
except Exception , e :
log_error ( ' Failed to save player seed for %s : %s ' % ( identity , str ( e ) ) )
link . send ( ' An error occured ' )
try :
ps = GetPlayerSeed ( link , ' dice ' )
2015-01-12 21:51:52 +00:00
except Exception , e :
2015-01-13 12:28:05 +00:00
log_error ( ' Failed to retrieve newly set player seed for %s : %s ' % ( identity , str ( e ) ) )
link . send ( ' An error occured ' )
return
link . send ( ' Your new player seed is: %s ' % ps )
2015-01-12 21:51:52 +00:00
2015-01-13 12:28:05 +00:00
def FairCheck ( link , cmd ) :
identity = link . identity ( )
2015-01-12 21:51:52 +00:00
try :
2015-01-13 12:28:05 +00:00
seed = GetServerSeed ( link , ' dice ' )
2015-01-12 21:51:52 +00:00
except Exception , e :
2015-01-13 12:28:05 +00:00
log_error ( ' Failed to get server seed for %s : %s ' % ( identity , str ( e ) ) )
link . send ( ' An error has occured ' )
2015-01-12 21:51:52 +00:00
return
try :
2015-01-13 12:28:05 +00:00
GenerateServerSeed ( link , ' dice ' )
2015-01-12 21:51:52 +00:00
except Exception , e :
2015-01-13 12:28:05 +00:00
log_error ( ' Failed to generate server seed for %s : %s ' % ( identity , str ( e ) ) )
link . send ( ' An error has occured ' )
2015-01-12 21:51:52 +00:00
return
2015-01-13 12:28:05 +00:00
link . send ( ' Your server seed was %s - it has now been reset; see !fair for details ' % str ( seed ) )
2015-01-12 21:51:52 +00:00
2015-01-13 12:28:05 +00:00
def Seeds ( link , cmd ) :
identity = link . identity ( )
2015-01-12 21:51:52 +00:00
try :
2015-01-13 12:28:05 +00:00
sh = GetServerSeedHash ( link , ' dice ' )
ps = GetPlayerSeed ( link , ' dice ' )
2015-01-12 21:51:52 +00:00
except Exception , e :
2015-01-13 12:28:05 +00:00
log_error ( ' Failed to get server seed for %s : %s ' % ( identity , str ( e ) ) )
link . send ( ' An error has occured ' )
2015-01-12 21:51:52 +00:00
return
2015-01-13 12:28:05 +00:00
link . send ( ' Your server seed hash is %s ' % str ( sh ) )
2015-01-12 21:51:52 +00:00
if ps == " " :
2015-01-20 18:00:02 +00:00
link . send ( ' You have not set a player seed ' )
2015-01-12 21:51:52 +00:00
else :
2015-01-13 12:28:05 +00:00
link . send ( ' Your player seed hash is %s ' % str ( ps ) )
2015-01-12 21:51:52 +00:00
2015-01-13 12:28:05 +00:00
def Fair ( link , cmd ) :
2015-01-20 17:18:15 +00:00
link . send_private ( " %s ' s dice betting is provably fair " % config . tipbot_name )
link . send_private ( " Your rolls are determined by three pieces of information: " )
link . send_private ( " - your server seed. You can see its hash with !seeds " )
link . send_private ( " - your player seed. Empty by default, you can set it with !playerseed " )
link . send_private ( " - the roll number, displayed with each bet you make " )
link . send_private ( " To verify past rolls were fair, use !faircheck " )
link . send_private ( " You will be given your server seed, and a new one will be generated " )
link . send_private ( " for future rolls. Then follow these steps: " )
link . send_private ( " Calculate the SHA-256 sum of serverseed:playerseed:rollnumber " )
link . send_private ( " Take the first 8 characters of this sum to make an hexadecimal " )
2015-01-31 13:01:03 +00:00
link . send_private ( " number, and divide it by 0xffffffff. You will end up with a number " )
2015-01-20 17:18:15 +00:00
link . send_private ( " between 0 and 1 which was your roll for that particular bet " )
link . send_private ( " See !faircode for Python code implementing this check " )
2015-01-12 21:51:52 +00:00
2015-01-13 12:28:05 +00:00
def FairCode ( link , cmd ) :
2015-01-20 17:18:15 +00:00
link . send_private ( " This Python 2 code takes the seeds and roll number and outputs the roll " )
link . send_private ( " for the corresponding game. Run it with three arguments: server seed, " )
link . send_private ( " player seed (use ' ' if you did not set any), and roll number. " )
2015-01-12 21:51:52 +00:00
2015-01-20 17:18:15 +00:00
link . send_private ( " import sys,hashlib,random " )
link . send_private ( " try: " )
link . send_private ( " s=hashlib.sha256(sys.argv[1]+ ' : ' +sys.argv[2]+ ' : ' +sys.argv[3]).hexdigest() " )
2015-01-31 13:01:03 +00:00
link . send_private ( " roll = float(long(s[0:8],base=16))/0xffffffff " )
2015-01-20 17:18:15 +00:00
link . send_private ( " print ' %.16g ' % r oll " )
link . send_private ( " except: " )
link . send_private ( " print ' need serverseed, playerseed, and roll number ' " )
2015-01-12 21:51:52 +00:00
2015-01-13 12:28:05 +00:00
def DiceHelp ( link ) :
2015-01-20 17:18:15 +00:00
link . send_private ( " The dice module is a provably fair %s dice betting game " % coinspecs . name )
link . send_private ( " Basic usage: !dice <amount> <multiplier> [over|under] " )
link . send_private ( " The goal is to get a roll under (or over, at your option) a target that depends " )
link . send_private ( " on your chosen profit multiplier (1 for even money) " )
link . send_private ( " See !fair and !faircode for a description of the provable fairness of the game " )
link . send_private ( " See !faircheck to get the server seed to check past rolls were fair " )
2015-01-12 21:51:52 +00:00
random . seed ( time . time ( ) )
RegisterModule ( {
' name ' : __name__ ,
' help ' : DiceHelp ,
} )
RegisterCommand ( {
' module ' : __name__ ,
' name ' : ' dice ' ,
2015-01-24 16:58:14 +00:00
' parms ' : ' <amount-in- %s > <multiplier> [over|under] ' % coinspecs . name ,
2015-01-12 21:51:52 +00:00
' function ' : Dice ,
' registered ' : True ,
2015-01-17 09:02:37 +00:00
' help ' : " play a dice game - house edge %.1f %% " % ( float ( config . dice_edge ) * 100 )
2015-01-12 21:51:52 +00:00
} )
RegisterCommand ( {
' module ' : __name__ ,
' name ' : ' stats ' ,
' parms ' : ' [<name>] ' ,
' function ' : GetDiceStats ,
' registered ' : True ,
' help ' : " displays your dice stats "
} )
RegisterCommand ( {
' module ' : __name__ ,
' name ' : ' resetstats ' ,
' parms ' : ' [<name>] ' ,
' function ' : ResetDiceStats ,
' registered ' : True ,
' help ' : " resets your dice stats "
} )
RegisterCommand ( {
' module ' : __name__ ,
' name ' : ' playerseed ' ,
' parms ' : ' <string> ' ,
' function ' : PlayerSeed ,
' registered ' : True ,
' help ' : " set a custom seed to use in the hash calculation "
} )
RegisterCommand ( {
' module ' : __name__ ,
' name ' : ' seeds ' ,
' function ' : Seeds ,
' registered ' : True ,
' help ' : " Show hash of your current server seed and your player seed "
} )
RegisterCommand ( {
' module ' : __name__ ,
' name ' : ' faircheck ' ,
' function ' : FairCheck ,
' registered ' : True ,
' help ' : " Check provably fair rolls "
} )
RegisterCommand ( {
' module ' : __name__ ,
' name ' : ' fair ' ,
' function ' : Fair ,
' help ' : " describe the provably fair dice game "
} )
RegisterCommand ( {
' module ' : __name__ ,
' name ' : ' faircode ' ,
' function ' : FairCode ,
' help ' : " Show sample Python code to check bet fairness "
} )