微服务架构 CI/CD 实战

简介: 微服务架构 CI/CD 实战

概述


现有 Demo,一项目某块业务拆分为如下几个模块

  • book-web 前端,采用 Vue MVVM,服务端 Thymeleaf SSR 渲染,友好 SEO MPA。服务端 路由,Spring MVC
  • book-gateway 微服务网关,Spring Cloud Gateway
  • book-service 接口提供方,Spring Cloud Alibaba Dubbo 服务消费方
  • book-server 数据承载方,Spring Cloud Alibaba Dubbo 服务提供方
  • book-common 实体,Dubbo 接口 jar
  • cloud-common 微服务公共 jar
  • common pom
  • school-parent 最上级 父工程 pom

本篇内容使用Jenkis实现自动构建。


common 继承关系

在项目根路径下执行,Maven 命令 mvn clean install -pl com.lab:book-common -am -Ptest,截取输出如下

[INFO] Reactor Summary for school_parent 0.0.1-SNAPSHOT:
[INFO]
[INFO] school_parent ...................................... SUCCESS [  0.761 s]
[INFO] common ............................................. SUCCESS [  0.083 s]
[INFO] cloud-common ....................................... SUCCESS [  6.430 s]
[INFO] book-common ........................................ SUCCESS [  3.236 s]
[INFO] ------------------------------------------------------------------------
[INFO] BUILD SUCCESS
[INFO] ------------------------------------------------------------------------
[INFO] Total time:  11.880 s
[INFO] Finished at: 2020-05-03T14:13:12+08:00
[INFO] ------------------------------------------------------------------------


相关中间件

docker 和 docker-compose 适合 测试环境使用,生产环境用 Kubernetes,相关内容本公众号都有,可以自行查阅。

root@jazz-pc:/opt# docker ps --format "table {{.Names}} ------------- {{.Image}}"
NAMES ------------- IMAGE
rmqbroker ------------- apacherocketmq/rocketmq:4.5.2-alpine
rmqnamesrv ------------- apacherocketmq/rocketmq:4.5.2-alpine
seata-server ------------- seataio/seata-server:1.2.0
nacos ------------- nacos/nacos-server:1.2.1
minio ------------- minio/minio
es ------------- elasticsearch:7.6.2
zookeeper ------------- zookeeper
mysql ------------- mysql:5.7
mongo ------------- mongo
redis ------------- redis

Alibaba Sentinel

nohup java -Dserver.port=8858 -Dsentinel.dashboard.auth.username=sentinel -Dsentinel.dashboard.auth.password=Aa123456 -jar sentinel-dashboard-1.7.2.jar >sentinel.log 2>&1 &


Maven 环境

root@jazz-pc:/opt# mvn -v
Apache Maven 3.6.3 (cecedd343002696d0abb50b32b541b8a6ba2883f)
Maven home: /usr/local/apache-maven-3.6.3
Java version: 11.0.7, vendor: Oracle Corporation, runtime: /usr/local/java/jdk-11.0.7
Default locale: zh_CN, platform encoding: UTF-8
OS name: "linux", version: "5.3.0-51-generic", arch: "amd64", family: "unix"

在settings.xml 文件中,添加国内镜像

<mirror>
  <id>aliyunmaven</id>
  <mirrorOf>*</mirrorOf>
  <name>阿里云公共仓库</name>
  <url>https://maven.aliyun.com/repository/public</url>
</mirror>

顶级父工程 pom.xml 添加,跳过Maven单元测试 编译和执行

<maven.test.skip>true</maven.test.skip>
<skipTests>true</skipTests>

如果有 Maven Nexus,则另外添加设置,install 替换为 deploy


Jenkins相关配置

启动Jenkis

nohup java -jar /usr/local/jenkins/jenkins.war --ajp13Port=-1 --httpPort=8086 >/usr/local/jenkins/jenkins.out 2>&1 &
  • Jenkins JDK 环境 >=1.8 and <=11 目前最新版本不支持 Java 14
  • Jenkins 安装一般会卡住 2 次,则是去国外下载插件数据缓慢的原因,kill -9 进程 ID,ps -ef | grep jenkins 找到进程 ID。
    第一次卡住修改 :/root/.jenkins/hudson.model.UpdateCenter.xml 文件里面链接内容为国内镜像地址 https://mirrors.tuna.tsinghua.edu.cn/jenkins/updates/update-center.json
    第二次卡住执行替换命令:
cd /root/.jenkins/updates
sed -i 's/http:\/\/updates.jenkins-ci.org\/download/https:\/\/mirrors.tuna.tsinghua.edu.cn\/jenkins/g' default.json && sed -i 's/http:\/\/www.google.com/https:\/\/www.baidu.com/g' default.json


Git or SVN

root@jazz-pc:/opt# which git
/usr/bin/git
root@jazz-pc:/opt# which svn
/usr/bin/svn


Jenkins CI/CD

  • Global Tool Configuration 设置好,JDK,Maven,Git
  • 新建 Item,Freestyle project,如下

image.png

Item book-common

  • 源码管理,选择 Git,填写 Repository URL 和 Credentials 设置 Jenkins 凭据
  • 构建 Execute shell
#!/bin/bash
cd ./schoolmvn clean install -pl com.lab:book-common -am -Ptest
echo 'install ok!'


Item book-web

  • 源码管理 略
  • Execute shell


#!/bin/bash
cd ./school/book-webmvn clean package -Ptest
echo 'package ok!'echo 'build start!'
service_name="book-web"service_prot=80
IID=$(docker images | grep "$service_name" | awk '{print $3}')echo "IID $IID"
if [ -n "$IID" ]then    echo "exist $service_name image,IID=$IID"
    docker rmi -f $service_name    echo "delete $service_name image"
    docker build -t $service_name .    echo "build $service_name image"else    echo "no exist $service_name image,build docker"
    docker build -t $service_name .    echo "build $service_name image"fi
CID=$(docker ps -a | grep "$service_name" | awk '{print $1}')echo "CID $CID"
if [ -n "$CID" ]then    echo "exist $service_name container,CID=$CID"
    docker stop $service_name    docker rm $service_nameelse    echo "no exist $service_name container"fi
docker run -d --name $service_name \-v /etc/localtime:/etc/localtime:ro \-v /etc/timezone:/etc/timezone:ro \-v /data/logs:/data/logs:rw \--net=host -p $service_prot:$service_prot $service_name


Item book-gateway

  • 源码管理和 Execute shell 略

Item book-service

  • Execute shell
#!/bin/bash
cd ./school/book-servicemvn clean package -Ptest
echo 'package ok!'echo 'build start!'
service_name="book-service"service_prot=30880
IID=$(docker images | grep "$service_name" | awk '{print $3}')echo "IID $IID"
if [ -n "$IID" ]then    echo "exist $service_name image,IID=$IID"
    docker rmi -f $service_name    echo "delete $service_name image"
    docker build --no-cache -t $service_name .    echo "build $service_name image"else    echo "no exist $service_name image,build docker"
    docker build -t $service_name .    echo "build $service_name image"fi
CID=$(docker ps -a | grep "$service_name" | awk '{print $1}')echo "CID $CID"
if [ -n "$CID" ]then    echo "exist $service_name container,CID=$CID"
    docker stop $service_name    docker rm $service_nameelse    echo "no exist $service_name container"fi
docker run -d --name $service_name \-e DUBBO_IP_TO_REGISTRY=192.168.1.6 \-e DUBBO_PORT_TO_REGISTRY=$service_prot \-e DUBBO_IP_TO_BIND=192.168.1.6 \-p $service_prot:$service_prot \-v /etc/localtime:/etc/localtime:ro \-v /etc/timezone:/etc/timezone:ro \-v /data/logs:/data/logs:rw \--net=host $service_name


Item book-server

  • Execute shell

#!/bin/bash
cd ./school/book-servermvn clean package -Ptest
echo 'package ok!'echo 'build start!'
service_name="book-server"service_prot=20880
IID=$(docker images | grep "$service_name" | awk '{print $3}')echo "IID $IID"
if [ -n "$IID" ]then    echo "exist $service_name image,IID=$IID"
    docker rmi -f $service_name    echo "delete $service_name image"
    docker build --no-cache -t $service_name .    echo "build $service_name image"else    echo "no exist $service_name image,build docker"
    docker build -t $service_name .    echo "build $service_name image"fi
CID=$(docker ps -a | grep "$service_name" | awk '{print $1}')echo "CID $CID"
if [ -n "$CID" ]then    echo "exist $service_name container,CID=$CID"
    docker stop $service_name    docker rm $service_nameelse    echo "no exist $service_name container"fi
docker run -d --name $service_name \-e DUBBO_IP_TO_REGISTRY=192.168.1.6 \-e DUBBO_PORT_TO_REGISTRY=$service_prot \-e DUBBO_IP_TO_BIND=192.168.1.6 \-p $service_prot:$service_prot \-v /etc/localtime:/etc/localtime:ro \-v /etc/timezone:/etc/timezone:ro \-v /data/logs:/data/logs:rw \--net=host \$service_name


Dockerfile

  • 位置和 pom.xml 路径平级,book-server 内容如下,其他略
FROM adoptopenjdk/openjdk11:jdk-11.0.7_10-alpine
VOLUME ["/tmp","/data/logs"]
COPY ./target/book-server-0.0.1-SNAPSHOT.jar book-server.jar
ENTRYPOINT ["java","-Djava.security.egd=file:/dev/./urandom","-jar","/book-server.jar","&"]


book-web 执行构建 Build Now


book-web 前端 浏览器访问

image.png

login.html 登录模板示例页面,调用 /api/login 接口,内容如下

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8"/>
    <title>理想生活上天猫,登录页面</title>
    <r th:insert="common/header::#headerApp"/>
</head>
<body>
<div id="app">
    <el-row :gutter="20">
        <el-col :span="24"> </el-col>
    </el-row>
    <el-row :gutter="20">
        <el-col :span="1"> </el-col>
        <el-col :span="4">
            <img alt="我是一只天猫" src="//img.alicdn.com/tfs/TB11ojWRXXXXXafaFXXXXXXXXXX-190-27.png">
        </el-col>
        <el-col :span="19"> </el-col>
    </el-row>
    <el-row :gutter="20">
        <el-col :span="15">
            <img alt="九寨沟" src="/images/JiuZhaiGou.jpg" width="800" height="490">
        </el-col>
        <el-col :span="7">
            <template>
                <el-tabs v-model="activeName" @tab-click="handleClick">
                    <el-tab-pane label="密码登录" name="first">
                        <el-form ref="form" :model="userForm" label-width="80px">
                            <el-row>
                                <el-input placeholder="会员名/邮箱/手机号" prefix-icon="el-icon-user" v-model="userForm.name"></el-input>
                            </el-row>
                            <el-row>
                                <el-input placeholder="请输入密码" prefix-icon="el-icon-lock" v-model="userForm.password" show-password></el-input>
                            </el-row>
                            <el-row>
                                <el-button @click="login" type="danger">登录</el-button>
                            </el-row>
                        </el-form>
                    </el-tab-pane>
                    <el-tab-pane label="短信登录" name="second">短信登录</el-tab-pane>
                </el-tabs>
            </template>
        </el-col>
        <el-col :span="2"> </el-col>
    </el-row>
</div>
<r th:replace="common/footer::.footerApp"/>
<script src="/js/request.js"></script>
<script>
    new Vue({
        el: '#app',
        data: {
            userForm:{}, activeName: 'first'
        },
        methods: {
            login() {
                request({
                    url: '/api/login',method: 'post',data: this.userForm,
                    headers: {
                        'content-type': 'application/json;charset=UTF-8',"token": ''
                    }
                }).then(response=>{
                    var res = response.data;
                    this.$message(res.message);
                })
            },
            handleClick(tab, event) {
                //console.log(tab, event);
            }
        }
    });
</script>
</body>
</html>


好了,各位朋友们,本期的内容到此就全部结束啦,能看到这里的同学都是优秀的同学,下一个升职加薪的就是你了!

相关实践学习
消息队列RocketMQ版:基础消息收发功能体验
本实验场景介绍消息队列RocketMQ版的基础消息收发功能,涵盖实例创建、Topic、Group资源创建以及消息收发体验等基础功能模块。
消息队列 MNS 入门课程
1、消息队列MNS简介 本节课介绍消息队列的MNS的基础概念 2、消息队列MNS特性 本节课介绍消息队列的MNS的主要特性 3、MNS的最佳实践及场景应用 本节课介绍消息队列的MNS的最佳实践及场景应用案例 4、手把手系列:消息队列MNS实操讲 本节课介绍消息队列的MNS的实际操作演示 5、动手实验:基于MNS,0基础轻松构建 Web Client 本节课带您一起基于MNS,0基础轻松构建 Web Client
目录
相关文章
|
19天前
|
Kubernetes Cloud Native Docker
云原生之旅:从容器到微服务的架构演变
【8月更文挑战第29天】在数字化时代的浪潮下,云原生技术以其灵活性、可扩展性和弹性管理成为企业数字化转型的关键。本文将通过浅显易懂的语言和生动的比喻,带领读者了解云原生的基本概念,探索容器化技术的奥秘,并深入微服务架构的世界。我们将一起见证代码如何转化为现实中的服务,实现快速迭代和高效部署。无论你是初学者还是有经验的开发者,这篇文章都会为你打开一扇通往云原生世界的大门。
|
20天前
|
负载均衡 应用服务中间件 持续交付
微服务架构下的Web服务器部署
【8月更文第28天】随着互联网应用的不断发展,传统的单体应用架构逐渐显露出其局限性,特别是在可扩展性和维护性方面。为了解决这些问题,微服务架构应运而生。微服务架构通过将应用程序分解成一系列小型、独立的服务来提高系统的灵活性和可维护性。本文将探讨如何在微服务架构中有效部署和管理Web服务器实例,并提供一些实际的代码示例。
51 0
|
14天前
|
Dubbo Java 应用服务中间件
微服务框架Dubbo环境部署实战
微服务框架Dubbo环境部署的实战指南,涵盖了Dubbo的概述、服务部署、以及Dubbo web管理页面的部署,旨在指导读者如何搭建和使用Dubbo框架。
67 17
微服务框架Dubbo环境部署实战
|
4天前
|
运维 Cloud Native Devops
云原生架构的崛起与实践云原生架构是一种通过容器化、微服务和DevOps等技术手段,帮助应用系统实现敏捷部署、弹性扩展和高效运维的技术理念。本文将探讨云原生的概念、核心技术以及其在企业中的应用实践,揭示云原生如何成为现代软件开发和运营的主流方式。##
云原生架构是现代IT领域的一场革命,它依托于容器化、微服务和DevOps等核心技术,旨在解决传统架构在应对复杂业务需求时的不足。通过采用云原生方法,企业可以实现敏捷部署、弹性扩展和高效运维,从而大幅提升开发效率和系统可靠性。本文详细阐述了云原生的核心概念、主要技术和实际应用案例,并探讨了企业在实施云原生过程中的挑战与解决方案。无论是正在转型的传统企业,还是寻求创新的互联网企业,云原生都提供了一条实现高效能、高灵活性和高可靠性的技术路径。 ##
12 3
|
8天前
|
监控 负载均衡 应用服务中间件
探索微服务架构下的API网关设计与实践
在数字化浪潮中,微服务架构以其灵活性和可扩展性成为企业IT架构的宠儿。本文将深入浅出地介绍微服务架构下API网关的关键作用,探讨其设计原则与实践要点,旨在帮助读者更好地理解和应用API网关,优化微服务间的通信效率和安全性,实现服务的高可用性和伸缩性。
26 3
|
12天前
|
存储 Java Maven
从零到微服务专家:用Micronaut框架轻松构建未来架构
【9月更文挑战第5天】在现代软件开发中,微服务架构因提升应用的可伸缩性和灵活性而广受欢迎。Micronaut 是一个轻量级的 Java 框架,适合构建微服务。本文介绍如何从零开始使用 Micronaut 搭建微服务架构,包括设置开发环境、创建 Maven 项目并添加 Micronaut 依赖,编写主类启动应用,以及添加控制器处理 HTTP 请求。通过示例代码展示如何实现简单的 “Hello, World!” 功能,并介绍如何通过添加更多依赖来扩展应用功能,如数据访问、验证和安全性等。Micronaut 的强大和灵活性使你能够快速构建复杂的微服务系统。
33 5
|
11天前
|
自然语言处理 Java 网络架构
解锁跨平台微服务新纪元:Micronaut与Kotlin联袂打造的多语言兼容服务——代码、教程、实战一次打包奉送!
【9月更文挑战第6天】Micronaut是一款轻量级、高性能的Java框架,适用于微服务开发。它支持Java、Groovy和Kotlin等多种语言,提供灵活的多语言开发环境。本文通过创建一个简单的多语言兼容服务,展示如何使用Micronaut及其注解驱动特性实现REST接口,并引入国际化支持。无论是个人项目还是企业应用,Micronaut都能提供高效、一致的开发体验,成为跨平台开发的利器。通过简单的配置和代码编写,即可实现多语言支持,展现其强大的跨平台优势。
24 2
|
12天前
|
运维 监控 持续交付
深入浅出:微服务架构的设计与实战
微服务,一个在软件开发领域如雷贯耳的名词,它代表着一种现代软件架构的风格。本文将通过浅显易懂的语言,带领读者从零开始了解微服务的概念、设计原则及其在实际项目中的运用。我们将一起探讨如何将一个庞大的单体应用拆分为灵活、独立、可扩展的微服务,并分享一些实践中的经验和技巧。无论你是初学者还是有一定经验的开发者,这篇文章都将为你提供新的视角和深入的理解。
39 3
|
20天前
|
消息中间件 Java 网络架构
AMQP与微服务架构的集成策略
【8月更文第28天】在微服务架构中,各个服务通常通过HTTP/REST、gRPC等协议进行交互。虽然这些方法在很多场景下工作得很好,但在需要高并发、低延迟或需要处理大量消息的情况下,传统的同步调用方式可能无法满足需求。此时,AMQP作为异步通信的一种标准协议,可以提供一种更为灵活和高效的消息传递机制。
21 1
|
17天前
|
前端开发 开发者 C#
WPF开发者必读:MVVM模式实战,轻松实现现代桌面应用架构,让你的代码更上一层楼!
【8月更文挑战第31天】在WPF应用程序开发中,MVVM(Model-View-ViewModel)模式通过分离应用程序的逻辑和界面,提高了代码的可维护性和可扩展性。本文介绍了MVVM模式的三个核心组件:Model(数据模型)、View(用户界面)和ViewModel(处理数据绑定和逻辑),并通过示例代码展示了如何在WPF项目中实现MVVM模式。通过这种方式,开发者可以构建更加高效和可扩展的桌面应用程序。
39 0