小试牛刀-Telebot区块链游戏机器人

本文涉及的产品
应用实时监控服务-可观测链路OpenTelemetry版,每月50GB免费额度
MSE Nacos/ZooKeeper 企业版试用,1600元额度,限量50份
容器镜像服务 ACR,镜像仓库100个 不限时长
简介: 本文章为记录自己开发基于区块链和Telebot实现的[石头、剪刀、布]游戏的过程,加深自己对区块链知识的理解和使用,加深对TeleBot依赖库的使用,同时希望可以帮助到有想实现相关功能的朋友.

 目录

1.编写目的

2.实现功能

2.1 Wallet功能

2.2  游戏功能

2.3  提出功能

2.4  辅助功能

3.功能实现详解

3.1 wallet功能

3.2 游戏功能

3.3 提出功能

3.4 辅助功能

4.测试视频


Welcome to Code Block's blog

本篇文章主要介绍了

[Telebot区块链游戏机器人]

❤博主广交技术好友,喜欢文章的可以关注一下❤

1.编写目的

       本文章为记录自己开发基于区块链和Telebot实现的[石头、剪刀、布]游戏的过程,加深自己对区块链知识的理解和使用,加深对TeleBot依赖库的使用,同时希望可以帮助到有想实现相关功能的朋友.    

2.实现功能

2.1 Wallet功能

      用户可以通过/create命令命令创建游戏wallet,同时可以输入地址或扫码向该wallet发送一定数量的游戏代币,使用/wallet命令可以显示当前游戏wallet内的剩余代币数量。

2.2  游戏功能

       用户可以将bot机器人添加到公开群组,同时在群组内发送/game命令创建具有奖励随机结果的游戏,使用不同的人向群组内回复该游戏/pk进行对战,获取对战结果并向游戏wallet发送设定的奖励游戏代币,用户可以使用/del删除当前已创建游戏。

2.3  提出功能

       用户可以通过/bind命令绑定外部wallet并进行代币提出.

2.4  辅助功能

       用户可以输入/rules查看游戏规则,输入/help命令查看机器人命令及解释。

3.功能实现详解

3.1 wallet功能

       用户通过/create命令命令创建游戏wallet可以参考我的博客生成solana公私钥,/wallet显示wallet信息是通过solana.py根据保存的用户公钥查询.

def getBalance(publicKey:str):
    solana_client = Client(rpc_url)
    #公钥转换
    pubkey=Pubkey.from_string(publicKey)
    tokenPublicKey=Pubkey.from_string(BOGGY_TOKEN_MINT)
    #获取SOL余额
    sol_balance = solana_client.get_balance(pubkey)
    #获取SPL代币余额
    token_account=solana_client.get_token_accounts_by_owner_json_parsed(pubkey,TokenAccountOpts(mint=tokenPublicKey))
    if noTokenAccount(token_account):
        #不存在代币账户时,则余额为0
        token_balance=0.0
    else:
        token_account_json=token_account.value[0].account.to_json()
        token_balance=json.loads(token_account_json)['data']['parsed']['info']['tokenAmount']['uiAmount']
    sol_balance=(sol_balance.value/10**9)
    return sol_balance,token_balance

image.gif

       同时为方便用户使用,这边会将用户公钥通过qrcode库转换为二维码供用户扫码,主要代码为:

import qrcode
from io import BytesIO
from PIL import Image
import qrcode.main
def generate_qr(data):
    # 生成普通二维码
    qr = qrcode.main.QRCode(
        version=1,
        box_size=10,
        border=4,
    )
    qr.add_data(data)
    qr.make(fit=True)
    qr_img = qr.make_image(fill_color="black", back_color="white")
    
    # 将二维码图片保存到 BytesIO 对象中
    img_io = BytesIO()
    qr_img.save(img_io, format='PNG')
    img_io.seek(0)
    
    return img_io

image.gif

实现效果:

image.gif 编辑

3.2 游戏功能

       游戏功能的设计思路是当用户在群组内输入/game时,根据(群组id+用户id+消息id)生成唯一的游戏id并通过json文件存储创建者生成的随机值信息.存储信息如下:

{
  "create_user_id": 5385955983,
  "create_username": "GameOfBoggy",
  "create_select": 0,
  "pk_user_id": 2038830708,
  "pk_username": "USERT1223",
  "pk_select": 1,
  "pk_select_hex": "4b32e3c83744655cd4ab5cc991a342c99f52c73fa83f2393a995d53baf7aeb42",
  "amount": 20,
  "winner": "USERT1223",
  "create_select_hex": "bd55d5aa6e461c63c811ff78bb00753c517eba47f086e563783a5e023ff342af",
  "timestamp": 1721092512.869742,
  "create_drand_hex": "e3b4d8b6af061ddc40449d87c57c06d93de8fa73dff0055a07cc8dadb047dd1e",
  "pk_drand_hex": "808391bb3761db7f5be8ba296e143154f043ecced8e7a693698c8490300fe34f"
}

image.gif

这里的是使用本地secrets库和调用远程的drand随机数链生成两个hash值(create_drand_hex和create_select_hex为创建者游戏结果计算hash,pk_select_hex和pk_drand_hex为pk者游戏结果计算hash,这些结果面会保存到链上以保证游戏结果公平性和随机性.),这两个hash值转换为整数相加并对3取余获得随机结果,以保证游戏的随机性和不可预测性:

调用远程drand API:

def get_drand_randomness():
    # Drand API endpoint
    url = "https://api.drand.sh/52db9ba70e0cc0f6eaf7803dd07447a1f5477735fd3f661792ba94600c84e971/public/latest"
    # Send request to Drand
    response = requests.get(url)
    if response.status_code == 200:
        # Parse the JSON response
        data = response.json()
        randomness = data['randomness']
        return randomness
    else:
        return secrets.token_hex()

image.gif

计算游戏随机结果:

def get_random_hex_int():
    select_hex=secrets.token_hex()
    drand_hex=get_drand_randomness()
    hex_int = int(select_hex, 16)
    drand_int = int(drand_hex,16)
    select=(drand_int+hex_int)%3
    return select,select_hex,drand_hex

image.gif

为保证游戏的美观性,这边使用图片+文字方式让机器人回复用户随机结果:

def random_game(bot,message,type,create_username):
    select,select_hex,drand_hex = get_random_hex_int()
    imgPath="./img/{}.png".format(img[select])
    msgText="<b>[{}] You random [{}]! @{}</b>".format(type,img[select],create_username)
    send_message=bot.send_photo(message.chat.id, open(imgPath, 'rb'), caption=msgText,parse_mode='HTML')
    return select,send_message,select_hex,drand_hex

image.gif

游戏奖励发放给获胜者为代币转移操作,实现代码如下(这里的draw_data即为计算出结果的hash值,将其备注添加链上):

#向绑定账户发送代币
def drawTokenAccount(sender_public_key:str,privateKey:str,draw_public_key:str,tokenAmount,draw_data):
    solana_client = Client(rpc_url)
    #发送者
    sender_pubkey = Pubkey.from_string(sender_public_key)
    #接收者
    draw_pubkey = Pubkey.from_string(draw_public_key)
    #Token代币地址
    token_mint_address = Pubkey.from_string(BOGGY_TOKEN_MINT)
    #发送者keypair
    sender_keypair=Keypair.from_base58_string(privateKey)
    
    try:
        #spl_client客户端
        source_token_account=get_associated_token_address(sender_pubkey,token_mint_address)
        dest_token_account=get_associated_token_address(draw_pubkey,token_mint_address)
        #交易
        transfer_instruction = transfer_checked(
            TransferCheckedParams(
                program_id=TOKEN_PROGRAM_ID,
                source=source_token_account,
                mint=token_mint_address,
                dest=dest_token_account,
                owner=sender_pubkey,
                amount=int(float(tokenAmount) * 1000000000),
                decimals=9,
            )
        )
        memo_instruction=create_memo(MemoParams(
            program_id=MEMO_PROGRAM_ID,
            signer=sender_pubkey,
            message=draw_data.encode('utf-8')
        ))
        # #获取最新的区块hash
        recent_blockhash_resp = solana_client.get_latest_blockhash()
        recent_blockhash=recent_blockhash_resp.value.blockhash
        # # 创建交易并添加转账指令
        transaction = Transaction()
        transaction.add(set_compute_unit_limit(200000))
        transaction.add(set_compute_unit_price(7500))
        transaction.add(transfer_instruction)
        transaction.add(memo_instruction)
        #设置最新区块hash
        transaction.recent_blockhash=recent_blockhash
        #设置手续费支付地址为发送者
        transaction.fee_payer=sender_pubkey
        #签名
        transaction.sign(sender_keypair)
        #发送交易
        response = solana_client.send_raw_transaction(transaction.serialize())
        #打印交易
        return response.value
    except Exception as e:
        print(f"Exception occurred: {str(e)}")
        return "error"

image.gif

实现效果:

/game 20 即创建了一个奖励为20代币的游戏,这里随机的是[剪刀]

image.gif 编辑

/pk pk者出了[布],所以创建者获得了奖励.

image.gif 编辑

我们可以点击按钮在链上查看奖励内容.

image.gif 编辑

可以看到这里包含了一个转移代币操作,是从pk者转移到创建者中的,同时包含了生成随机结果的hash值以保证游戏的公开和公平性。

3.3 提出功能

       用户输入/bind 绑定自己的链上地址后,提出功能即为代币转移操作,主要实现代码如下,这边直接根据用户的ID查到用户的创建的游戏wallet公钥私钥并进行转移:

def drawTokenFromUserId(send_user_id,draw_user_id,gameId,draw_data):
    publicKey,privateKey=get_account(send_user_id)
    drawPublicKey=getPublicKey(draw_user_id)
    amount=get_game_amount(gameId)
    tx=drawTokenAccount(publicKey,privateKey,drawPublicKey,amount,draw_data)
    return amount,tx

image.gif

3.4 辅助功能

       辅助功能即为telebot消息的回复,实现代码如下:

/rules:

from telebot import types
def handle_rules(bot, message):
    # 处理 /start 命令
    markup = types.InlineKeyboardMarkup()
    item1 = types.InlineKeyboardButton("BOGGY GROUP",url="https://t.me/BoggyCoin")
    markup.add(item1)
    welcome_message=(
        "<b>"
        "1.Before starting the game, you need to create(/create) a game wallet and transfer a small amount of sol and BOGGY tokens inward\n\n"
        "2.You can send '/game [amount]' create a game(default amount:500),and the Pker reply /pk with [GAME](need enough sol and BOGGY)\n\n"
        "3.Waiting for the results, the winner will receive the bonus set by the game creator\n\n"
        "4.The [Scissors] will win [Paper],[Parer] will win [Rock],[Rock] will win [Scissors]\n\n"
        "5.You Can at /wallet,draw you all token in you bind wallet,you can send '/bind [address]' bind you wallet\n\n"
        "[Create With #BOGGY]"
        "</b>"
    )
    bot.send_photo(message.chat.id,open("./img/rules.jpeg","rb"),welcome_message,parse_mode='HTML',reply_markup=markup)
def register_handlers(bot):
    bot.message_handler(commands=['rules'])(lambda message: handle_rules(bot, message))

image.gif

/help:

from telebot import types
def handle_help(bot, message):
    markup = types.InlineKeyboardMarkup()
    item1 = types.InlineKeyboardButton("BOGGY GROUP",url="https://t.me/BoggyCoin")
    markup.add(item1)
    help_text = (
        "<b>Welcome!</b>\n" 
        "<b>Here are the available commands:</b>\n"
        "<b>[/create]         Create you game wallet [DM*]</b>\n"
        "<b>[/bind]           Bind you draw wallet [DM*]</b> \n"
        "<b>[/wallet]         Show you wallet Info and Draw [DM*]</b>\n\n"
        "<b>[/game]           Create Game with amount</b>\n"
        "<b>[/pk]             Reply the Game Message and Pk it</b>\n"
        "<b>[/rules]          View more detailed game rules</b>\n\n"
        "<b>The [DM*] need DM Bot</b>"
    )
    
    bot.send_photo(message.chat.id,open("./img/help.jpeg","rb"),help_text, parse_mode='HTML',reply_markup=markup)
def register_handlers(bot):
    bot.message_handler(commands=['help'])(lambda message: handle_help(bot, message))

image.gif

4.测试视频

video_2024-07-16_11-48-41

image.gif

 

感谢您的关注和收藏!!!!!!

image.gif 编辑

目录
相关文章
|
4月前
|
jenkins Java 持续交付
使用Jenkins完成springboot项目快速更新
本文介绍了使用Jenkins和WinSW实现SpringBoot项目自动化部署的完整流程。首先讲解了Jenkins作为持续集成工具的作用,然后详细说明了环境准备步骤:包括JDK版本管理、WinSW服务配置(含XML文件修改)以及bat启动脚本编写。重点演示了Jenkins的项目配置方法,包括源码管理设置和构建步骤中的Windows批处理命令调用。通过这套方案,开发者只需推送代码到Git仓库,即可触发Jenkins自动完成项目构建、服务重启等全流程,显著提升部署效率。文章还提到IDEA的Jenkins插件可进
155 1
|
2月前
|
Oracle Java 关系型数据库
SpringBoot从0-1集成Graalvm
本文介绍如何使用GraalVM将SpringBoot应用打包为原生可执行文件并构建Docker镜像。相比传统JAR包,原生镜像启动更快、体积更小,提升部署效率,适合现代云原生环境。
276 10
|
4月前
|
算法 关系型数据库 Java
Springboot集成PostGIS完成路径规划
因为公司里需要做关于林区防火方面的项目,需要完成着火后山区路径的导航,但.....某德的功能似乎只能到达山区的边上,后边的路就需要自己完成导航了。搞了一个周终于有所效果了,也遇见了很多的坑,在此记录一下,希望以后不要踩坑。需要上述的环境才能进行路径导航,环境的搭建可以参阅
121 5
|
4月前
|
安全 算法 区块链
openssl生成证书
本文章是记录openssl命令生成私钥、证书签名请求、CA证书的命令和相关参数的解释。其中包含了各参数的名称、作用、技术细节和安全建议。
96 1
|
4月前
|
存储 JavaScript 区块链
小试牛刀-SOL链创建Token
最近需要编写SOL合约进行SPL Token的转移,因为在测试网上需要自己部署测试Token,同时为了更加美观,Token需携带metadata数据(对名称、头像等)进行定义.在此对创建过程进行记录,希望帮助到有需要实现相关功能的朋友.
85 1
|
4月前
|
存储 前端开发 测试技术
小试牛刀-区块链代币锁仓合约实战
记录一下自己在开发代币合约中的过程,加深自己对合约功能的理解,在后续的学习过程中可以进行资料查阅,以及帮助有这方面开发要求或想学习的朋友进行更方便的入门。
100 1
|
4月前
|
IDE Java 开发工具
IntelliJ IDEA 使用技巧与插件推荐
IntelliJ IDEA 是一个功能强大、扩展性丰富的开发工具。通过掌握常用的快捷键和技巧,结合合适的插件,可以大幅提升你的开发效率。
165 2
|
4月前
|
缓存 前端开发 Java
SpringBoot 实现动态菜单功能完整指南
本文介绍了一个动态菜单系统的实现方案,涵盖数据库设计、SpringBoot后端实现、Vue前端展示及权限控制等内容,适用于中后台系统的权限管理。
316 1
|
4月前
|
NoSQL Java 区块链
Springboot应用开发:配置类整理
在使用SpringBoot进行应用开发中,通常需要编写很多的配置类,这些配置类在很大程度上提高了应用的开发速度.其中就包括线程池、数据库连接池、缓存的相关配置类。在这里我将经常用到的配置类进行整理和总结。
208 1
|
4月前
|
存储 NoSQL 区块链
开源:LMDB 操作工具:lmcmd
本文介绍了 LMDB(一种高效的键值存储数据库)和基于 Python 开发的命令行工具 `lmcmd`。由于 LMDB 使用二进制文件存储,管理和调试不便,因此开发了 `lmcmd`,提供了类似 Redis 的命令行操作界面,支持数据库操作、数据导入导出和查找等功能。文章涵盖了 `lmcmd` 的安装、连接数据库和常用命令(如 `set`、`get`、`export` 等)示例。最后强调了开源工具的价值,鼓励用户反馈和改进。
101 1