资源简介 2023年全国职业院校技能大赛高职组“区块链技术应用”赛项赛卷(5卷)任务书参赛队编号:背景描述电子签章可实现与纸质文件盖章操作相似的可视效果,以保障数据来源的真实性、数据完整性以及签名人行为的不可否认性。传统的电子签章系统是基于中心化的,也就是数据是集中存储在中心数据库中,这就导致传统电子签章使用记录存在被篡改、伪造的风险。而区块链电子签章系统在传统电子签章系统的基础上,可以借助于区块链技术,用不可篡改、不可抵赖的方式记录电子签章的整个流转过程,如领取、使用、查询等,从而解决传统电子签章系统存在的潜在风险。某市政务部分拟开发一款区块链电子签章系统,该区块链电子签章系统包含为印章使用者、监管、平台运营方的管理功能,场景开放平台,印章底层引擎,和区块链基座。场景开放平台面向各类政务系统与第三方电子签名服务商提供电子签章相关的开放能力。场景开放平台对上提供相关的基础能力,对下连接区块链基座。模块一:区块链产品方案设计及系统运维(35分)选手完成本模块的任务后,将任务中设计结果、运行代码、运行结果等截图粘贴至客户端桌面【区块链技术应用赛\重命名为工位号\模块一提交结果.docx】中对应的任务序号下。任务1-1:区块链产品需求分析与方案设计本任务需要依据项目背景完成需求分析与方案设计,具体要求如下:1.依据给定区块链电子印章系统的业务架构图,对考题进行业务分析,尽可能多的去考虑一个业务系统所需要的模块,使用Visio或思维导图工具展现本系统的基本设计概念和处理流程,要求分为区块链电子印章业务平台和支撑平台两个部分;2.根据电子印章系统设计,在系统中分别包括用户和签章管理功能。根据描述,设计区块链电子签章系统的总体功能概览图;3.根据电子印章系统设计,请设计区块链电子签章系统通过智能合约确保用户隐私和数据保密的方案;4.根据电子印章系统设计,请设计区块链电子签章系统中的身份验证和授权管理具体方案。任务1-2:区块链系统部署与运维围绕电子签章区块链平台部署与运维需求,进行项目相关系统、节点以及管理工具的部署工作。通过通过监控工具完成对网络、节点服务的监控。最终利用业务需求规范,完成系统日志、网络参数、节点服务等系统结构维护。1. 登陆Linux服务器,安装并部署下图所示的单机、四机构、三群组、八节点的星形组网拓扑区块链系统;2. 登陆Linux服务器,安装并部署区块链系统控制台,检查部署控制台是否正常运行;3. 登录Linux服务器,部署区块链管理前置平台;4. 登陆Linux服务器,使用终端生成新的节点,并且将该节点加入对应群组然后启动节点。子任务1-2-1: 登陆Linux服务器,安装并部署下图所示的单机、四机构、三群组、八节点的星形组网拓扑区块链系统,具体工作内容如下:搭建部署多群组联盟链并启动所有节点;通过命令验证区块链节点进程运行状况;通过命令验证区块链连接状态和共识状态日志输出。子任务1-2-2:登陆Linux服务器,安装并部署区块链系统控制台,检查部署控制台是否正常运行,具体工作内容如下:解压控制台安装包并拷贝配置文件到当前目录;配置控制台证书;启动控制台;验证控制台是否正常运行。子任务1-2-3:登录Linux服务器,部署区块链管理前置平台,具体内容如下解压平台操作对应安装包;配置秘钥文件并检查前置平台启动情况;查看日志以及查看中间件进程;检查进程端口以及访问服务。子任务1-2-4:登陆Linux服务器,使用终端生成新的节点,并且将该节点加入对应群组然后启动节点,具体内容如下:生成新节点,修改新节点配置;启动新节点,并查看节点的nodeid;将新节点作为观察节点加入群组1当中,并检查是否加入成功。任务1-3:区块链系统测试设计对区块链系统的测试流程;结合实际业务需求,调用部署的智能合约中进行系统测试、性能测试等;根据业务需求,分析并且修复给定智能合约中的安全漏洞。利用模拟业务和测试工具来完成对区块链系统服务数据的测试。1. 基于给定脚本完成区块链管理平台部署以及结果验证,最后将执行结果截图保存。实现区块链管理平台部署;实现管理平台启动情况验证;验证身份认证进程启动情况验证和浏览器验证。智能合约安全漏洞测试。有如下智能合约:pragma solidity ^0.8.3;contract EtherGame {uint public targetAmount = 7 ether;address public winner;function deposit() public payable {require(msg.value == 1 ether, "You can only send 1 Ether");uint balance = address(this).balance;require(balance <= targetAmount, "Game is over");if (balance == targetAmount) {winner = msg.sender;}}function claimReward() public {require(msg.sender == winner, "Not winner");(bool sent, ) = msg.sender.call{value: address(this).balance}("");require(sent, "Failed to send Ether");}}contract Attack {EtherGame etherGame;constructor(EtherGame _etherGame) {etherGame = EtherGame(_etherGame);}function attack() public payable {address payable addr = payable(address(etherGame));selfdestruct(addr);}}分析智能合约中存在问题,并说明危害;创建新的智能合约,修复其中问题;说明修复内容并测试其智能合约。模块二:智能合约开发与测试(30分)选手完成本模块的任务后,将任务中设计结果、运行代码、运行结果等截图粘贴至客户端桌面【区块链技术应用赛\重命名为工位号\模块二提交结果.docx】中对应的任务序号下。任务2-1:智能合约设计根据区块链电子签章应用需求分析和方案设计文档的描述,编写该区块链产品的智能合约功能需求文档,以及设计该智能合约UML图,具体要求如下:1.编写区块链电子签章智能合约功能需求文档根据区块链电子签章产品项目背景和需求分析,编写该区块链电子签章产品的智能合约功能需求文档。2.完成区块链电子签章智能合约的设计图设计区块链电子签章智能合约接口,画出区块链电子签章智能合约的角色UML用例图,以及画出区块链电子签章智能合约关系的时序图。任务2-2:智能合约开发使用Solidity语言完成智能合约开发、部署和调用,要求如下:1.个人签章信息接口编码(1)编写个人签章智能合约的实体接口,完成实体通用数据的初始化,实现签章和用户实体信息上链的功能;struct SealInfo {//①姓名//②身份证//③印章数据//④创建印章时间戳}(2)编写签章信息上链的接口,实现Seal合约的构造函数;//@dev构造函数// @param _name姓名// @param _cardId身份证号码// @param _data 个人印章数据// @param _datetime 账户注册时间constructor(string _name,string _cardId,string _data,string _datetime) public {//①用户姓名//②用户身份ID//③用户数据//④时间戳isUsed = true;}(3)基于给定的智能合约代码以及注释,完成ElectronicSeal合约判断多人签章文件编号是否存在的函数。//@dev 判断多人签名文件编号是否存在//@param _code 签章文件编号//@return bool true:存在 false:不存在function isExistMultiCode(string _code) view internal returns(bool) {//①遍历所有多人签章文件编号并与传入签章文件编号参数进行比较//②返回不存在}2.电子印章接口编码(8分)(1)基于给定的智能合约代码以及注释,完成ElectronicSeal合约获取多人签章信息函数(3分)//@dev 获取多人签章信息//@param _code 签章文件编号//@return (签章文件摘要、多人签章发起人账户地址、签章文件编号、签章人数、签章账户地址列表)function getMultiSingInfo(string _code) public view onlyPromoterOrSinger returns(string,address,string,uint,address[] memory) {//①获取多人签章合约对象实例//②返回多人签章信息}(2)基于给定的智能合约代码以及注释,完成ElectronicSeal合约多人签章函数(5分)//@dev 多人签章//@param _address 签章人账户地址//@param _hash 签章文件摘要//@param _datatime 签章时间戳function doMultiSign(address _address,string _hash,string _datetime) public onlyPromoter {//①判断传入的签章人账户地址不能为空//②获取多人签章合约对象实例//③获取当前等待签章账户地址//④判断合约调用者是否为等待签章人账户地址//⑤进行签章}合约编译、部署和调用(1)解决代码错误和警告,正确编译并部署合约,成功获取部署的合约地址和abi;(2)调用区块链电子签章智能合约的接口,完整验证业务流程。任务2-3:智能合约测试根据已完成的智能合约,针对开发功能开展相关合约测试工作,具体工作内容如下:生成测试文件,在Remix中的合约测试目录tests中生成ElectionSeal_Test.sol的测试文件编译并部署合约,配置项目的ElectionSeal_Test.sol测试脚本文件。注释测试文件中自动生成的测试函数checkSuccess()、checkSuccess2()、checkFailure()、checkSenderAndValue()相关代码,运行测试按钮基于remix 的测试项目,将checkSuccess()、checkSuccess2()、checkFailure()、checkSenderAndValue()相关代码注释掉并进行相关测试;实例化在测试文件中定义测试ElectronicSeal合约变量并在beforeAll()函数中进行实例化。创建测试函数测试印章账户数量基于remix 的测试项目,补全位于test文件夹中的ElectionSeal_Test.sol合约测试文件,添加测试用例,补全测试印章账户数量函数。创建测试函数测试印章账户签章基于remix 的测试项目,补全位于test文件夹中ElectionSeal_Test.sol合约测试文件,添加测试用例,创建测试函数测试印章账户签章。 模块三:区块链应用系统开发(30分)任务3-1:区块链电子签章前端功能开发1.请基于前端系统的开发模板,在注册页面register.html文件中添加对应的代码逻辑,实现对前端系统的注册功能,并测试功能完整性,示例页面如下;题目的具体要求如下:注册身份证号码的数据实现双向绑定注册按钮的事件绑定将用户输入的注册信息向后端系统发送注册请求register.html代码片段1:代码片段2:代码片段3:methods: {submit() {if (this.user.username == '') {alert("用户名不能为空")return}if (this.user.password == '') {alert("密码不能为空")return}if (this.user.chainAccount == '') {alert("区块链账户地址不能为空")return}if (this.user.name == '') {alert("姓名不能为空")return}if (this.user.cardId == '') {alert("身份证号码不能为空")return}if (this.user.phone == '') {alert("电话号码不能为空")return}选手填写部分}}2.请基于前端系统的开发模板,在登录页面login.html文件中添加对应的代码逻辑,实现对前端系统的登录功能,并测试功能完整性,示例页面如下:本题目的具体要求如下:登录按钮的事件绑定用户输入的登录用户名和密码通过双向绑定存放在 user 中登录成功页面跳转到main.html 页面本地保存登录状态信息(token)login.html代码片段1:代码片段2://创建Vue实例new Vue({el: '#app', //el用于指定当前Vue实例为哪个容器服务,值通常为css选择器字符串data: { //data中用于存储数据,数据供el所指定的容器去使用,值我们暂时先写成一个对象。user: {选手填写部分},tip: false,msg: ""},methods: {submit() {if (this.user.username == '') {this.msg = "用户名不能为空"this.tip = truesetTimeout(() => {this.tip = false}, 1000)return}if (this.user.password == '') {this.msg = "密码不能为空"this.tip = truesetTimeout(() => {this.tip = false}, 1000)return}axios.post("http://localhost/user/login", this.user).then(response => {if (response.data.resultCode == 200) {this.msg = "登录成功"this.tip = truesetTimeout(() => {选手填写部分}, 1000)选手填写部分} else {this.msg = "登录失败"this.tip = truesetTimeout(() => {this.tip = false}, 1000)}})},}})3.请基于前端系统的开发模板,在个人印章页面myseal.html文件中添加对应的页面代码逻辑,实现对前端系统的个人印章功能,并测试功能完整性,示例页面如下:具体要求如下:注销按钮的事件绑定注销时清除本地保存的登录状态信息(token)发送请求到后端系统携带本地保存的登录状态信息(token)接收后端系统发送过来的印章图片信息myseal.html代码片段1:注销代码片段2:methods:{logout() {选手填写部分}},代码片段3:// 请求发前拦截,header中添加 tokenaxios.interceptors.request.use(config => {config.headers.Authorization =选手填写部分return config})代码片段4:axios.get("http://localhost/user/seal").then(response => {if (response.data.resultCode == 200) {this.seal =选手填写部分}})4.请基于前端系统的开发模板,请在文件签章页面signature.html文件中添加对应的页面代码逻辑,实现对前端系统的文件签章业务功能,并测试功能完整性,示例页面如下:具体要求如下:创建canvas画布实例向后端系统发送签章请求url向后端系统获取个人印章请求url接收后端系统发送过来的印章图片信息signature.html代码片段1:signature() {if (this.status == 0) {this.msg = "请上传文档"this.tip = truesetTimeout(() => {this.tip = false}, 1000)return}this.status = 2var canvas;canvas =选手填写部分canvas.width = $("#poster").width()canvas.height = $("#poster").height()var context = canvas.getContext("2d");context.rect(0, 0, canvas.width, canvas.height);context.fillStyle = "#fff";context.fill();代码片段2:uploadSeal() {if (this.status != 2) {this.msg = "文档未上传或未签章"this.tip = truesetTimeout(() => {this.tip = false}, 1000)return}var posterImg = document.getElementById('poster');var imgBase64 = posterImg.getAttribute("src");axios.post("选手填写部分",{"filename": this.filename,"imgBase64": imgBase64,}).then(response => {if (response.data.resultCode == 200) {this.msg = "签章文档上链成功"this.tip = truesetTimeout(() => {this.tip = falsewindow.location.href = "main.html";}, 1000)} else {this.msg = "签章文档上链失败"this.tip = truesetTimeout(() => {this.tip = false}, 1000)}})}代码片段3:mounted() {// 请求发前拦截,header中添加 tokenaxios.interceptors.request.use(request => {request.headers.Authorization = localStorage.getItem("Authorization")return request})//响应拦截器(认证授权)axios.interceptors.response.use(response => {if (response.data.resultCode == 401) {//跳转页面window.location.href = "login.html";}return response})//TODO:【5-1-4-3】请补充向服务器发送获取个人印章请求代码axios.get("选手填写部分").then(response => {if (response.data.resultCode == 200) {this.seal =选手填写部分}})}5.请基于前端系统的开发模板,在文件验章页面verity.html文件中添加对应的页面代码逻辑,实现对前端系统的文件验章业务功能,并测试功能完整性,示例页面如下:具体要求如下:将验章结果数据显示在页面中向后端系统发送验章请求并携带验章文件数据verity.html代码片段1:aria-hidden="true">×签章信息姓名:选手填写部分身份证号码:选手填写部分区块链账户地址:选手填写部分文件名称:选手填写部分文件编码:选手填写部分签章时间:选手填写部分代码片段2:async verify(event) { //同步发送 axios 请求(获取数据后弹出模态框)if (this.status == 0) {this.msg = "请上传文档"this.tip = trueevent.stopPropagation() //阻止事件冒泡(防止弹出模态框)setTimeout(() => {this.tip = false}, 1000)return}await axios.post("选手填写部分", {"imgBase64":this.contract}).then(response => {if (response.data.resultCode == 200) {this.verifyInfo = response.data.data}})},任务3-2:区块链应用系统后端开发1.请基于后端系统的开发模板,在userService.java文件中补充对应的代码逻辑,完成后端代码逻辑,实现对后端系统的注册功能,并测试功能完整性,题目具体要求如下:根据用户注册信息中的用户名和账户地址从数据库中查询账户信息是否存在将用户密码进行md5加密操作将电子印章字节数组转为字符串将用户注册信息写入数据库汇总userService.java代码片段1:public boolean register(RegisterBean registerBean) {//校验用户名或账户地址是否重复QueryWrapper queryWrapper = new QueryWrapper<>();queryWrapper.eq("username", registerBean.getUsername()).or().eq("chain_account",registerBean.getChainAccount());UserEntity userOne =选手填写部分if (userOne != null) {log.info("注册失败:用户名或账户地址重复");return false;}//拷贝对象UserEntity userEntity = new UserEntity();BeanUtils.copyProperties(registerBean, userEntity);userEntity.setPassword(选手填写部分);//md5加密密码//生成印章(Base64)byte[] buffer = Seal.getSealImg(registerBean.getUsername(), registerBean.getName());Encoder encode = Base64.getEncoder();String sealImg =选手填写部分userEntity.setSealImg(sealImg);//用户注册信息上链List funcParams = new ArrayList<>();funcParams.add(registerBean.getChainAccount());funcParams.add(registerBean.getName());funcParams.add(registerBean.getCardId());funcParams.add(String.valueOf(System.currentTimeMillis()));JSONObject res = HttpUtils.writeContract("addSealAccount", funcParams);if (res == null) {log.info("注册失败:用户注册信息上链失败");return false;}int result =选手填写部分return result > 0 true : false;}2.请基于后端系统的开发模板,补充代码完成对应的后端代码逻辑,实现后端系统的登录功能,并测试功能完整性,具体要求如下:登录成功向前端系统返回登录状态信息(token)添加登录的用户名和密码两个数据库查询条件根据登录的用户名和密码到数据库中进行查询操作UserController.java代码片段1:public AjaxResult login(@RequestBody LoginBean loginBean) {log.info(loginBean.toString());if (userService.isLogin(loginBean)) {//创建String token = TokenUtil.sign(loginBean.getUsername(),String.valueOf(System.currentTimeMillis()));AjaxResult result = new AjaxResult<>(200, "login ok");result.setData(选手填写部分);return result;} else {return new AjaxResult(403,"login fail");}}UserService.java代码片段1:public boolean isLogin(LoginBean loginBean) {QueryWrapper queryWrapper = new QueryWrapper<>();loginBean.setPassword(SecureUtil.md5(loginBean.getPassword()));//md5加密密码//TODO:请补充 queryWrapper 添加用户名和密码两个数据库查询条件代码queryWrapper.eq(选手填写部分).eq(选手填写部分);//TODO:请补充数据库查询登录用户信息代码UserEntity userEntity =选手填写部分return userEntity != null true : false;}3.请基于后端系统的开发模板,补充代码完成对应的后端代码逻辑,实现对后端系统的账户信息查看功能,并测试功能完整性,具体要求如下:从前端系统发送的请求头中获取登录状态信息(token)根据登录状态信息(token)获取登录用户名根据用户名查询数据库将查询结果数据封装到RegisterBean 中UserController.java代码片段1:@ApiOperation(value = "用户信息",notes = "获取用户详细信息")@GetMapping("/user/info")public AjaxResult getUserInfo(HttpServletRequest request) {String token =选手填写部分String username =选手填写部分RegisterBean registerBean = userService.getUser(username);AjaxResult result = new AjaxResult<>(200, "ok");result.setData(registerBean);return result;}UserService.java代码片段1:public RegisterBean getUser(String username) {QueryWrapper queryWrapper = new QueryWrapper<>();queryWrapper.eq("username", username);UserEntity userEntity =选手填写部分RegisterBean registerBean = new RegisterBean();选手填写部分return registerBean;}4.请基于后端系统的开发模板,补充代码完成对应的后端代码逻辑,实现对后端系统的文件签章功能,并测试功能完整性,具体要求如下:验证登录状态信息(token)获取签章文件的UUID编码获取验章文件摘要(hash)将验章文件信息写入智能合约中TokenInterceptor.java代码片段1:public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {if ("OPTIONS".equals(request.getMethod())) {response.setStatus(HttpServletResponse.SC_OK);return true;}response.setCharacterEncoding("utf-8");String token = request.getHeader("Authorization");log.info("token:"+token);if (token != null) {boolean result =选手填写部分if (result) {log.info("通过拦截器");return true;}}response.setCharacterEncoding("UTF-8");response.setContentType("application/json; charset=utf-8");PrintWriter out = null;try {JSONObject json = new JSONObject();json.put("message", "认证失败,未通过拦截器");json.put("resultCode", "401");response.getWriter().append(json.toJSONString());log.info("认证失败,未通过拦截器");} catch (Exception e) {e.printStackTrace();response.sendError(500);return false;}return false;}SealService.java代码片段1:public boolean signature(ContractBean contractBean, String chainAccount) {//获取签名文档uuid编码String code =选手填写部分SealEntity sealEntity = new SealEntity();BeanUtils.copyProperties(contractBean, sealEntity);String[] str = contractBean.getFilename().split("\\.");sealEntity.setFilename(str[0]);sealEntity.setType(str[1]);sealEntity.setCode(code);//签章记录上链try {//获取印章文档Hash值MessageDigest messageDigest =选手填写部分byte[] bytes = messageDigest.digest(contractBean.getImgBase64().getBytes("UTF-8"));String hash = Hex.toHexString(bytes);log.info("hash:" + hash);List funcParams = new ArrayList<>();funcParams.add(chainAccount);funcParams.add(hash);funcParams.add(code);funcParams.add(String.valueOf(System.currentTimeMillis()));JSONObject res =选手填写部分if (res == null) {log.info("印章记录上链失败");}} catch (Exception e) {e.printStackTrace();}int result = sealMapper.insert(sealEntity);return result > 0 true : false;}5.请基于后端系统的开发模板,补充代码完成对应的后端代码逻辑,实现对后端系统的文件验章功能,并测试功能完整性,具体要求如下:调用验章方法进行验章读取合约获取区块链账户信息读取合约查询区块链验章结果SealController.java代码片段1:public AjaxResult verify(@RequestBody ImgBase64Bean imgBase64Bean, HttpServletRequest request) {String imgBase64 = imgBase64Bean.getImgBase64();String token = request.getHeader("Authorization");String username = TokenUtil.getLoginName(token);RegisterBean registerBean = userService.getUser(username);VerifyBean verifyBean =选手填写部分if (verifyBean == null){return new AjaxResult(403,"fail");}AjaxResult result = new AjaxResult<>(200,"ok");result.setData(verifyBean);return result;}SealService.java代码片段1:public VerifyBean verify(String chainAccount, String imgBase64) {VerifyBean verifyBean = new VerifyBean();verifyBean.setChainAccount(chainAccount);//获取账户信息List params = new ArrayList();params.add(chainAccount);JSONArray accountInfo =选手填写部分if (accountInfo == null) {log.info("获取区块链账户信息失败");return null;}List list = JSONArray.parseArray(accountInfo.toJSONString(), String.class);String name = list.get(0);String cardId = list.get(1);verifyBean.setName(name);verifyBean.setCardId(cardId);try {//获取印章文档Hash值MessageDigest messageDigest = MessageDigest.getInstance("SHA-256");byte[] bytes = messageDigest.digest(imgBase64.getBytes("UTF-8"));String hash = Hex.toHexString(bytes);log.info("hash:"+hash);//获取印章信息List funcParams = new ArrayList();funcParams.add(chainAccount);funcParams.add(hash);JSONArray sealInfo =选手填写部分if (sealInfo == null) {log.info("获取区块链印章信息失败");return null;} 展开更多...... 收起↑ 资源预览