dfinity开发——动态创建Canister与Canister的控制器管理

2023-10-14 13:40

本文主要是介绍dfinity开发——动态创建Canister与Canister的控制器管理,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

这是dfinity中动态创建Canister例子,就是在A.mo 的actor中动态创建B.mo中的actor类。

项目

1.首先创建一个项目

dfx new CanisterMGT

2.启动项目

cd CanisterMGT
dfx start --clean

3.在项目下创建types.mo 和 sub.mo文件
在这里插入图片描述

4.types.mo是canister所需要的类型和属性,添加如下代码:

module{public type canister_settings ={freezing_threshold : ?Nat;controllers : ?[Principal];memory_allocation : ?Nat;compute_allocation : ?Nat;};public type definite_canister_settings = {freezing_threshold : Nat;controllers : [Principal];memory_allocation : Nat;compute_allocation : Nat;};public type user_id = Principal;public type wasm_module = [Nat8];public type canister_id = Principal;public type ICActor = actor {canister_status : shared { canister_id : canister_id } -> async {// status : { #stopped; #stopping; #running };// memory_size : Nat;// cycles : Nat;settings : { controllers : [Principal] };//module_hash : ?[Nat8];};create_canister : shared { settings : ?canister_settings } -> async {canister_id : canister_id;};delete_canister : shared { canister_id : canister_id } -> async ();deposit_cycles : shared { canister_id : canister_id } -> async ();install_code : shared {arg : [Nat8];wasm_module : wasm_module;mode : { #reinstall; #upgrade; #install };canister_id : canister_id;} -> async ();provisional_create_canister_with_cycles : shared {settings : ?canister_settings;amount : ?Nat;} -> async { canister_id : canister_id };provisional_top_up_canister : shared {canister_id : canister_id;amount : Nat;} -> async ();raw_rand : shared () -> async [Nat8];start_canister : shared { canister_id : canister_id } -> async ();stop_canister : shared { canister_id : canister_id } -> async ();uninstall_code : shared { canister_id : canister_id } -> async ();update_settings : shared {canister_id : Principal;settings : canister_settings;} -> async ();};
}

5.sub.mo是接下要在main.mo里面创建的canistet,sub.mo里面的代码就是它自身要实现的代码,但cycleBalance() 和wallet_receive()这两个属性是必须要添加在里面的,有必要也添加下canister的升级代码。

import Cycles "mo:base/ExperimentalCycles"; 
import Nat "mo:base/Nat";shared({caller}) actor class Sub(installer : Principal) = this
{public query({caller}) func cycleBalance() : async Nat{Cycles.balance()};public shared({caller}) func wallet_receive() : async Nat {Cycles.accept(Cycles.available())};public shared query ({caller}) func whoami() : async Principal {caller};stable var counter = 0;public query func get() : async Nat{return counter;};public func inc() : async () {counter += 1;};
}

6.main.mo是创建canister的代码

import Types "./types";
import Cycles "mo:base/ExperimentalCycles"; 
import RBT "mo:base/RBTree";
import Nat "mo:base/Nat";
import Principal "mo:base/Principal";
import Sub "./sub";
import Array "mo:base/Array";actor Self
{public type canister_info = {canister_id : Principal;id:Nat;};//获取当前pidpublic shared query ({caller}) func whoami() : async Principal {caller};//获取当前canister idpublic shared query ({caller}) func getCanisterId() : async Principal{Principal.fromActor(Self);};//记录创建的canister个数private stable var subs_index : Nat = 0;private let subs = RBT.RBTree<Nat, Principal>(Nat.compare); //存储创建的sub canisterprivate let IC : Types.ICActor = actor "aaaaa-aa"; private let CYCLE_LIMIT = 50; //创建canister要分配的cycle//动态创建canister的函数,返回创建的canisister idpublic shared({caller}) func createSub() : async canister_info{Cycles.add(CYCLE_LIMIT);let sub = await Sub.Sub(caller);let principal = Principal.fromActor(sub);await IC.update_settings({canister_id = principal;settings = {freezing_threshold = ?20;controllers = ?[principal]; //控制器是它自身memory_allocation = ?0;compute_allocation = ?0;}});subs.put(b_index, principal);subs_index += 1;let canister_info = {canister_id = principal;id = subs_index;}};//获取canister的控制器public func get_controllers(canister_id : Principal) : async [Principal] {let status = await IC.canister_status({ canister_id = canister_id });return status.settings.controllers;};
};

7.运行项目

dfx deploy

8.调用创建canister的接口

dfx canister call CanisterMGT createSub

动态创建第一个canister

(base) dalao@dalaodeMacBook-Pro CanisterMGT % dfx canister call CanisterMGT createSub
(record { id = 1 : nat; canister_id = principal "r7inp-6aaaa-aaaaa-aaabq-cai" })

9.现在来访问新创建的这个canister里面的属性
查看cycles,这50个是刚刚转进去的

(base) dalao@dalaodeMacBook-Pro CanisterMGT % dfx canister call r7inp-6aaaa-aaaaa-aaabq-cai cycleBalance
(50 : nat)

计数

base) dalao@dalaodeMacBook-Pro CanisterMGT % dfx canister call r7inp-6aaaa-aaaaa-aaabq-cai get         
(0 : nat)
(base) dalao@dalaodeMacBook-Pro CanisterMGT % dfx canister call r7inp-6aaaa-aaaaa-aaabq-cai inc
()
(base) dalao@dalaodeMacBook-Pro CanisterMGT % dfx canister call r7inp-6aaaa-aaaaa-aaabq-cai get
(1 : nat)
(base) dalao@dalaodeMacBook-Pro CanisterMGT % dfx canister call r7inp-6aaaa-aaaaa-aaabq-cai inc
()
(base) dalao@dalaodeMacBook-Pro CanisterMGT % dfx canister call r7inp-6aaaa-aaaaa-aaabq-cai get
(2 : nat)

然后再创建一个canister

(base) dalao@dalaodeMacBook-Pro CanisterMGT % dfx canister call CanisterMGT createSub                   
(record { id = 2 : nat; canister_id = principal "rkp4c-7iaaa-aaaaa-aaaca-cai" })

访问里面的属性

(base) dalao@dalaodeMacBook-Pro CanisterMGT % dfx canister call rkp4c-7iaaa-aaaaa-aaaca-cai get
(0 : nat)
(base) dalao@dalaodeMacBook-Pro CanisterMGT % dfx canister call rkp4c-7iaaa-aaaaa-aaaca-cai inc
()
(base) dalao@dalaodeMacBook-Pro CanisterMGT % dfx canister call rkp4c-7iaaa-aaaaa-aaaca-cai get
(1 : nat)
(base) dalao@dalaodeMacBook-Pro CanisterMGT % 

10.查看控制状态

如果canister当前只有一个控制器,它是拒绝外部访问它的控制器

(base) dalao@dalaodeMacBook-Pro CanisterMGT % dfx canister call CanisterMGT get_controllers '(principal "r7inp-6aaaa-aaaaa-aaabq-cai")'
Error: The Replica returned an error: code 4, message: "Only the controllers of the canister r7inp-6aaaa-aaaaa-aaabq-cai can control it.
Canister's controllers: r7inp-6aaaa-aaaaa-aaabq-cai
Sender's ID: rrkah-fqaaa-aaaaa-aaaaq-cai"

11.创建canister分配过去的cycle记得回收,我在测试的时创建一堆canister之后,退出环境之后才记得没有回收cycle,现在还在找办法找回来。

这篇关于dfinity开发——动态创建Canister与Canister的控制器管理的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/210873

相关文章

详解如何在SpringBoot控制器中处理用户数据

《详解如何在SpringBoot控制器中处理用户数据》在SpringBoot应用开发中,控制器(Controller)扮演着至关重要的角色,它负责接收用户请求、处理数据并返回响应,本文将深入浅出地讲解... 目录一、获取请求参数1.1 获取查询参数1.2 获取路径参数二、处理表单提交2.1 处理表单数据三、

一文教你如何解决Python开发总是import出错的问题

《一文教你如何解决Python开发总是import出错的问题》经常朋友碰到Python开发的过程中import包报错的问题,所以本文将和大家介绍一下可编辑安装(EditableInstall)模式,可... 目录摘要1. 可编辑安装(Editable Install)模式到底在解决什么问题?2. 原理3.

Python+PyQt5开发一个Windows电脑启动项管理神器

《Python+PyQt5开发一个Windows电脑启动项管理神器》:本文主要介绍如何使用PyQt5开发一款颜值与功能并存的Windows启动项管理工具,不仅能查看/删除现有启动项,还能智能添加新... 目录开篇:为什么我们需要启动项管理工具功能全景图核心技术解析1. Windows注册表操作2. 启动文件

使用Python开发Markdown兼容公式格式转换工具

《使用Python开发Markdown兼容公式格式转换工具》在技术写作中我们经常遇到公式格式问题,例如MathML无法显示,LaTeX格式错乱等,所以本文我们将使用Python开发Markdown兼容... 目录一、工具背景二、环境配置(Windows 10/11)1. 创建conda环境2. 获取XSLT

Android开发环境配置避坑指南

《Android开发环境配置避坑指南》本文主要介绍了Android开发环境配置过程中遇到的问题及解决方案,包括VPN注意事项、工具版本统一、Gerrit邮箱配置、Git拉取和提交代码、MergevsR... 目录网络环境:VPN 注意事项工具版本统一:android Studio & JDKGerrit的邮

Python开发文字版随机事件游戏的项目实例

《Python开发文字版随机事件游戏的项目实例》随机事件游戏是一种通过生成不可预测的事件来增强游戏体验的类型,在这篇博文中,我们将使用Python开发一款文字版随机事件游戏,通过这个项目,读者不仅能够... 目录项目概述2.1 游戏概念2.2 游戏特色2.3 目标玩家群体技术选择与环境准备3.1 开发环境3

gradle第三方Jar包依赖统一管理方式

《gradle第三方Jar包依赖统一管理方式》:本文主要介绍gradle第三方Jar包依赖统一管理方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录背景实现1.顶层模块build.gradle添加依赖管理插件2.顶层模块build.gradle添加所有管理依赖包

基于Python打造一个智能单词管理神器

《基于Python打造一个智能单词管理神器》这篇文章主要为大家详细介绍了如何使用Python打造一个智能单词管理神器,从查询到导出的一站式解决,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 项目概述:为什么需要这个工具2. 环境搭建与快速入门2.1 环境要求2.2 首次运行配置3. 核心功能使用指

Go语言开发实现查询IP信息的MCP服务器

《Go语言开发实现查询IP信息的MCP服务器》随着MCP的快速普及和广泛应用,MCP服务器也层出不穷,本文将详细介绍如何在Go语言中使用go-mcp库来开发一个查询IP信息的MCP... 目录前言mcp-ip-geo 服务器目录结构说明查询 IP 信息功能实现工具实现工具管理查询单个 IP 信息工具的实现服

HTML5中的Microdata与历史记录管理详解

《HTML5中的Microdata与历史记录管理详解》Microdata作为HTML5新增的一个特性,它允许开发者在HTML文档中添加更多的语义信息,以便于搜索引擎和浏览器更好地理解页面内容,本文将探... 目录html5中的Mijscrodata与历史记录管理背景简介html5中的Microdata使用M