DOCKER 学习笔记5 Springboot+nginx+mysql 容器编排
前言
在上节的内容中,我们已经通过一个简单的实例,将Docker-compose 进行了实际的应用。这一小节中。我们将通过学习和了解,着重认识容器的编排,上一节只算是一个小小的测试。在这一节中。我们将用最常见的技术。
- Springboot 作为后端应用
- Nginx 作为反向代理
- Mysql 持久化数据
Springboot 后端应用
引入JPA支持,以及MySQL的驱动
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
</dependency>
配置JPA 的基本属性
spring.jpa.properties.hibernate.hbm2ddl.auto=update
spring.jpa.properties.hibernate.dialect=org.hibernate.dialect.MySQL5InnoDBDialect
spring.jpa.show-sql=true
定义控制器
通过定义这样一个简单的控制器,访问/index
后就往数据库插入一条访问记录。
@RestController
@RequestMapping
public class UserController {
@Autowired
private UserRep userRep;
@GetMapping("/index")
public String index(HttpServletRequest request) {
UserEntity userEntity = new UserEntity();
userEntity.setName("guest");
userEntity.setCreateTime(new Date());
userEntity.setIp(request.getRemoteAddr());
userEntity.setStatus(1);
userRep.save(userEntity);
return "hello-docker-nginx";
}
}
实例代码:
自定义镜像
是否记得上次我们在使用 Dockerfile
自定义镜像?请参照:
这里我就不重复写了,差不多都一样的。
Dockerfile
还是放置到 src/docker/
下,不管放置到哪儿,只要你YML里面配置了一样的位置即可。
## 依赖基础镜像 maven 3.5 这个镜像里面已经包含有java
FROM maven:3.5-jdk-8
这是可能会有疑惑?为什么看不到 RUN
CMD
命令了,因为我们在后面将这些命令都指定到 docker-compose.yml
文件里面了。
配置NGINX
version: "3.0"
services:
nginx:
container_name: my-nginx
image: nginx:1.13
restart: always
ports:
- 8080:80
- 8443:443
volumes:
- ./nginx/conf.d:/etc/nginx/conf.d
简单说下一下:
version
这个就简单了,指明 docker-compose
文件的语法信息。当然,这个版本信息也不能忽视,从官网的一张表里面说明你的docker 版本与之对应的yml 文件的版本。需要查询自己docker 的版本后填写。
我这里是1.13.1
[root@mrclinux ~]# docker -v
Docker version 1.13.1, build 4ef4b30/1.13.1
container_name
这个就字面意思了,容器的名称
image
指定容器的构建镜像
restart
restart: "no"
restart: always
restart: on-failure
restart: unless-stopped
No 是默认的重新启动策略,在任何情况下都不会重新启动容器。
always,若没有启动成功,则容器始终重新启动。 (常用)
如果退出代码指示出现 on-failure 错误,则启动失败策略重新启动容器。
ports
将宿主主机上的端口与容器服务端口进行绑定,比如 8080:80
就是将宿主主机的8080端口绑定到这个服务的80端口
volumes
挂载主机路径或命名卷,指定为服务的子选项。
就意思说:把指定的路径挂载到服务当中。在这个例子里,
./nginx/conf.d:/etc/nginx/conf.d
就是将本文件夹./nginx/conf.d
映射到(挂载)到容器的/etc/nginx/conf.d
就是把NGINX 的配置文件夹给映射出来了。方便我们修改。
配置转发
server {
listen 80;
charset utf-8;
access_log off;
location / {
proxy_set_header Host $host:$server_port;
proxy_set_header X-Forwarded-Host $server_name;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_pass http://app:8080;
}
}
这个就简单了。没啥可说的,proxy_pass http://app:8080;
主要一下,因为app 这个服务和nginx 不在一个服务内,所以需要通过服务名的方式进行转发。
配置MYSQL
mysql:
container_name: my-mysql
image: mysql:8.0
environment:
MYSQL_DATABASE: demo
MYSQL_ROOT_PASSWORD: root
MYSQL_ROOT_HOST: '%'
ports:
- "3306:3306"
volumes:
- /root/database/mysql:/var/lib/mysql
restart: always
environment
环境变量,这个主要用于设置一些比如数据库的名称、以及用户密码等信息。
详见MYSQL 环境变量 https://hub.docker.com/_/mysql?tab=description
MYSQL_ROOT_PASSWORD
这个变量是强制性的,它指定将为 MySQL root 超级用户帐户设置的密码。
MYSQL_DATABASE
此变量是可选的,并允许您指定要在映像启动时创建的数据库的名称。
注意点
- 因为容器停止后,其里面的数据也会消失,这是我们所不能接受的,所以,将MYSQL 里面用来存储数据库的路径映射出来,映射到本机上我们所熟知的位置,这样就好了。
mkdir -p /root/database/mysql/
创建本地路径/root/database/mysql:/var/lib/mysql
组成映射关系。
配置后台Springboot
app:
container_name: my-app
build: .
working_dir: /app
volumes:
- ./:/app
- ~/.m2:/root/.m2
expose:
- "8080"
depends_on:
- nginx
- mysql
command: mvn clean spring-boot:run -Dspring-boot.run.profiles=docker
build
指定自定义镜像文件 Dockerfile
的目录,用于构建镜像
working_dir
指定功能目录
expose
在不将端口发布到主机的情况下公开端口——它们只能被链接的服务访问。 只能指定内部端口。
depends_on
启动顺序,意思是需要先启动 nginx mysql 而后启动本应用。
command
容器启动执行命令进行重写,其实就是将这个JAR包运行起来。
注意
~/.m2:/root/.m2
其实就是把宿主主机的Maven仓库映射到镜像内。
./:/app
其实就是将当前目录挂载到容器内 /app 下
运行测试
## 克隆仓库到服务器
git clone https://gitee.com/mrc1999/springboot-nginx-mysql-docker-compose.git
cd springboot-nginx-mysql-docker-compose
## 运行docker-compose 进行编排
docker-compose up
通过运行UP命令进行容器的编排与部署,观察打印的内容是否正确,若没有错误,则出现以下部分内容。
my-app | Downloaded from central: https://repo.maven.apache.org/maven2/org/apache/maven/maven-model/3.3.9/maven-model-3.3.9.jar (164 kB at 155 kB/s)
my-app |
my-app | . ____ _ __ _ _
my-app | /\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \
my-app | ( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
my-app | \\/ ___)| |_)| | | | | || (_| | ) ) ) )
my-app | ' |____| .__|_| |_|_| |_\__, | / / / /
my-app | =========|_|==============|___/=/_/_/_/
my-app | :: Spring Boot :: (v2.1.6.RELEASE)
my-app |
my-app | 2020-02-10 03:39:14.574 INFO 1 --- [ main] com.example.demo.DemoApplication : Starting DemoApplication on f27ac469d127 with PID 1 (/app/target/classes started by root in /app)
my-app | 2020-02-10 03:39:14.588 INFO 1 --- [ main] com.example.demo.DemoApplication : The following profiles are active: docker
Springboot 启动正常,因为这个时候是直接运行容器服务的,我们可以尝试使用 CTRL+C 进行关闭,而后通过后台的方式运行
后台运行
docker-compose up -d
[root@mrclinux springboot-nginx-mysql-docker-compose]# docker-compose up -d
my-nginx is up-to-date
Starting my-mysql ... done
Starting my-app ... done
检查容器运行情况
docker-compose ps
查看当前编排容器的运行情况
[root@mrclinux springboot-nginx-mysql-docker-compose]# docker-compose ps
Name Command State Ports
-----------------------------------------------------------------------------------------------
my-app /usr/local/bin/mvn-entrypo ... Up 8080/tcp
my-mysql docker-entrypoint.sh mysqld Up 0.0.0.0:3306->3306/tcp, 33060/tcp
my-nginx nginx -g daemon off; Up 0.0.0.0:8443->443/tcp, 0.0.0.0:8080->80/tcp
docker ps
通过运行 docker-ps
后发现,所有的镜像也已经存在。
[root@mrclinux springboot-nginx-mysql-docker-compose]# docker ps
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
f27ac469d127 springboot-nginx-mysql-docker-compose_app "/usr/local/bin/mv..." 6 minutes ago Up 31 seconds 8080/tcp my-app
5c0e28096c35 mysql:8.0 "docker-entrypoint..." 6 minutes ago Up 32 seconds 0.0.0.0:3306->3306/tcp, 33060/tcp my-mysql
2565244279b4 nginx:1.13 "nginx -g 'daemon ..." 6 minutes ago Up 30 seconds 0.0.0.0:8080->80/tcp, 0.0.0.0:8443->443/tcp my-nginx
docker-compose down
停止容器并且移除容器
[root@mrclinux springboot-nginx-mysql-docker-compose]# docker-compose down
Stopping my-app ... done
Stopping my-mysql ... done
Stopping my-nginx ... done
Removing my-app ... done
Removing my-mysql ... done
Removing my-nginx ... done
Removing network springboot-nginx-mysql-docker-compose_default
尝试发现
MYSQL 数据库持久化 ?
我们通过使用 volumes
将宿主主机的一个文件夹挂载到 my-mysql
容器下的 /var/lib/mysql
目录,那我们产生的数据呢?是否正常保存了么?
[root@mrclinux springboot-nginx-mysql-docker-compose]# ls /root/database/mysql/
auto.cnf binlog.000003 ca.pem demo ib_logfile0 #innodb_temp performance_schema server-cert.pem undo_001
binlog.000001 binlog.index client-cert.pem ib_buffer_pool ib_logfile1 mysql private_key.pem server-key.pem undo_002
binlog.000002 ca-key.pem client-key.pem ibdata1 ibtmp1 mysql.ibd public_key.pem sys
通过发现后得知,宿主主机的目录下已经产生了demo 的一个数据库。我们的挂载没有问题,下次还是从这里面读取即可。所以持久化数据没有问题。
working_dir 有啥用?
在容器运行的时候,我们尝试使用 docker exec -it my-app bash
使用命令行连接容器后,我们会发现
[root@mrclinux springboot-nginx-mysql-docker-compose]# docker exec -it my-app bash
root@f27ac469d127:/app#
root@f27ac469d127:/app#
root@f27ac469d127:/app# ls
Dockerfile docker-compose.yml maven mvnw mvnw.cmd nginx pom.xml src target
原来通过 working_dir 其实就是将当前目录的所有文件挂载到容器 /app 下。里面存在的文件,均是我通过映射拿进去的文件。
参考以及小结
通过这一节学习,基本上容器部署这些主流的MYSQL 以及NGINX 等已经全部没有任何问题了。学习到了卷的挂载。以及MYSQL 持久化数据的方式
要是遇到一些新的命令或者语法,再来记录吧~
参考
DOCKER 官网 https://docs.docker.com/compose/compose-file/
DOCKER 官网 https://docs.docker.com/storage/volumes/
DOCKER 官网 https://docs.docker.com/compose/gettingstarted/
纯洁的微笑 https://www.cnblogs.com/ityouknow/p/8661644.html
码云示例
https://gitee.com/mrc1999/springboot-nginx-mysql-docker-compose
DOCKER 学习笔记5 Springboot+nginx+mysql 容器编排的更多相关文章
- DOCKER 学习笔记4 认识DockerCompose 多容器编排
前言 通过上一节的学习,学会了如何在Linux 环境下搭建Docker并且部署Springboot 项目,并且成功的跑了起来,当然,在生产环境中,不只是需要一个后端的Web 项目,还需要比如 Ngin ...
- Docker学习笔记之--.Net Core应用容器通过网桥连接Redis容器(环境:centos7)
上节演示通过应用容器连接sql server容器,连接:Docker学习笔记之--.Net Core项目容器连接mssql容器(环境:centos7) 本节演示安装 redis容器,通过网桥连接 先决 ...
- Docker学习笔记之--.Net Core项目容器连接mssql容器(环境:centos7)
前一节演示在docker中安装mssql,地址:Docker学习笔记之--安装mssql(Sql Server)并使用Navicat连接测试(环境:centos7) 本节演示 .Net Core项目容 ...
- docker学习-lnmp+redis之搭建mysql容器服务
一. 前期准备工作,创建配置文件目录,log文件目录,数据库DATA和WEB站点目录[root@T1 ~]# mkdir -p /lnmp/conf/{mysql,nginx,php} /lnmp/l ...
- Docker学习笔记_04 Rancher的部署安装(编排选用K8S)
原文地址:http://dbase.cc/2018/01/12/docker/04_rancher的部署安装/ 为什么要使用Rancher Rancher是一个开源的企业级容器管理平台.通过Ranch ...
- Docker学习笔记4: Docker-Compose—简化复杂容器应用的利器
本文转载自http://www.tuicool.com/articles/AnIVJn. 因Python语言,个人也没学过,不是太熟悉,这篇文章的代码格式排版不准确爆了很多错,让我走了好多坑,不过还是 ...
- Docker学习笔记之运行和管理容器
0x00 概述 容器是基于容器技术所建立和运行的轻量级应用运行环境,它是 Docker 封装和管理应用程序或微服务的“集装箱”.在 Docker 中,容器算是最核心的部分了,掌握容器的操作也是 Doc ...
- Docker学习笔记 - 在运行中的容器内启动新进程
docker psdoker top dc1 # 容器情况# 在运行中的容器内启动新进程docker exec [-d] [-i] [-t] 容器名 [command] [args]docker ex ...
- Docker学习笔记之-部署.Net Core 3.1项目到Docker容器,并使用Nginx反向代理(CentOS7)(一)
上一节演示如何安装Docker,链接:Docker学习笔记之-在CentOS中安装Docker 本节演示 将.net core 3.1 部署到docker容器当中,并使用 Nginx反向代理,部署平台 ...
随机推荐
- 使用ASP.NET Core 3.x 构建 RESTful API - 4.2 过滤和搜索
向Web API传递参数 数据可以通过多种方式来传给API. Binding Source Attributes 会告诉 Model 的绑定引擎从哪里找到绑定源. 共有以下六种 Binding Sou ...
- 「CF242E」XOR on Segment 解题报告
题面 长度为\(n\)的数列,现有两种操作: 1.区间异或操作 2.区间求和操作 对于每个查询,输出答案 思路: 线段树+二进制拆位 线段树区间修改一般使用的都是懒标记的方法,但是对于异或,懒标记的方 ...
- 物理ceph集群+K8s
前提条件 在Ceph为k8s创建一个pool ceph osd pool create k8s 128 创建admin用户 ceph auth get-or-create client.admin m ...
- schedule of 2016-10-17~2016-10-23(Monday~Sunday)——1st semester of 2nd Grade
most important things to do 1.joint phd preparations 2.journal paper to write 3.solid fundamental kn ...
- lucene&tantivy对比
写入对比每个路径下都只能有一个IndexWriter负责写入,通过writer.lock实现.不同:lucene可以多个线程共享一个IndexWriter,每个线程负责写一个segment,从addD ...
- 【转】DB2数据库编目的概念以及对其的正确解析
此文章主要向大家描述的是DB2数据库编目的概念以及对DB2数据库编目的概念的正确理解,在DB2中编目(catalog)这个单词看似很难理解,我自己当初在学习DB2数据库的时候也常常被这个编目搞的很不明 ...
- Scala与Mongodb实践3-----运算环境的搭建
目的:使的在IDEA中编辑代码,令代码实现mongodb运算,且转换较为便捷 由实验2可知,运算环境的搭建亦需要对数据进行存储和计算,故需要实现类型转换,所以在实验2的基础上搭建环境. 由菜鸟教程可得 ...
- python类型检查和类型转换
类型检查 type()用来检查值的类型 (整型.浮点型.布尔值.字符串.空值) 该函数会将检查的结果作为返回值返回,可以通过变量来接收函数的返回值 print(type(1)) # <class ...
- React16源码解读:揭秘ReactDOM.render
引言 在上一篇文章中我们通过create-react-app脚手架快速搭建了一个简单的示例,并基于该示例讲解了在类组件中React.Component和React.PureComponent背后的实现 ...
- java中implements和extends的区别
1,extends是继承某个类的,可以使用某个类的方法,也可以重写父类的方法. 2,implements是用于实现类接口,可以实现一个或多个类的接口,接口的方法一般为空的,所以必须重写这一个或多个的方 ...