docker安装nginx
-
docker服务做成了随宿主机自启
systemctl enable docker
systemctl daemon-reload -
构建容器的时候,–restart=always
容器随docker服务的启动而启动虚拟机关机重启的话:首先是docker服务随虚拟机启动,容器又随docker服务的启动而启动
虚拟机启动,容器就自动启动了
构建java坏境的容器,部署java后台项目
- 上传文件到/zking/java目录(oa.jar Dockerfile start.sh)
mkdir -p /zking/java
rz
- 启动容器
cd /zking/java
sh start.sh
第一步:停止oa容器
第二步:删除oa容器
第三步:删除oa的镜像
第四步:构建自己的oa镜像
第五步:启动 jar 包
nginx是什么(见手绘图)
Nginx也是一款服务器,我们常用它做如:反向代理、负载均衡、动态与静态资源的分离的工作,
反向代理:反向代理(Reverse Proxy)方式是指以代理服务器来接受internet上的连接请求,然后将请求转发给内部网络上的服务器,并将从服务器上得到的结果返回给internet上请求连接的客户端,此时代理服务器对外就表现为一个服务器。
负载均衡:目前大多数的网站都会采用负载均衡手段来针对目前用户的指数级增长来减少对单点服务器的负载压力,比如目前我们拥有3台真实服务器,我们需要根据相应策略决定什么样的用户请求分配到哪个真实服务器,比如按照轮询的方式,用户请求挨个到达代理服务器,此时代理服务器按照第一个请求转发至第一台真实服务器,第二个请求转发到第二个服务器上,依次类推,这样可以防止大量的用户请求全部访问到同一台物理机上,单点物理机的性能始终有限的,当然这可能对服务器数据访问时候造成事务性的失效,在Web方面可能造成Session访问的问题,这不在本文讨论方面内
Nginx+tomcat是目前主流的java web部署架构
正向代理服务器
反向代理服务器
nginx访问流程图
安装nginx
-
搜索nginx镜像
docker search nginx
-
拉取镜像
docker pull nginx
-
在宿主机中创建挂载目录(见之前直接在虚拟机里面安装nginx后,会生成一些文件)
mkdir -p /zking/nginx/{conf,conf.d,html,log}
可以理解为:
mkdir /zking/nginx/conf
mkdir /zking/nginx/conf.d
mkdir /zking/nginx/html
mkdir /zking/nginx/log
注1:因为在nginx镜像中没有安装vi或vim编辑器(安装又太麻烦了),无法编辑nginx配置文件,
所以直接通过数据卷挂载上去更加方便 -
根据nginx镜像创建nginx容器
docker run
–name mynginx01
-d -p 80:80
nginx:latest等价:docker run --name mynginx01 -d -p 80:80 nginx:latest
注1:(测试nginx的安装与访问,并没有发布任何的java或vue项目,最后此容器要删除的)
-
查看本地的容器,可以看到nginx容器已经创建成功了
docker ps -a
-
测试nginx是否安装成功
打开浏览器输入:http://192.168.148.129( http://宿主机ip:80),正常情况下会显示nginx的欢迎页面
重新构建nginx容器用于发布vue项目
- 上传nginx.conf配置文件到宿主机/zking/nginx/conf目录下
注1:构建容器前上传配置文件到/zking/nginx/conf目录下,否则构建nginx容器会报错
nginx.conf讲解:
1. 所有配置必须以“;”结尾
2. nginx发布java项目
nginx+tomcat实现反向代理及均衡
upstream tomcats{
#TODO:172.17.0.3是docker容器的IP
server 172.17.0.3:8080 weight=1;#服务器配置 weight是权重的意思,权重越大,分配的概率越大。
server 172.17.0.4:8080 weight=2;
}
location ^~/api/ {
proxy_pass http://tomcats;
proxy_redirect default;
}
nginx.conf
#工作进程的个数,一般与计算机的cpu核数一致
worker_processes 1;
events {
#单个进程最大连接数(最大连接数=连接数*进程数)
worker_connections 1024;
}
http {
include mime.types;#文件扩展名与文件类型映射表
default_type application/octet-stream;#默认文件类型
sendfile on;#开启高效文件传输模式,sendfile指令指定nginx是否调用sendfile函数来输出文件,对于普通应用设为 on,如果用来进行下载等应用磁盘IO重负载应用,可设置为off,以平衡磁盘与网络I/O处理速度,降低系统的负载。注意:如果图片显示不正常把这个改成off。
keepalive_timeout 65; #长连接超时时间,单位是秒
gzip on;#启用Gizp压缩
#服务器的集群
upstream tomcats { #服务器集群名字
#TODO:172.17.0.3是docker容器的IP
server 172.17.0.3:8080 weight=1;#服务器配置 weight是权重的意思,权重越大,分配的概率越大。
server 172.17.0.4:8080 weight=2;
}
#当前的Nginx的配置
server {
listen 80;#监听80端口,可以改成其他端口
server_name localhost;#当前服务的域名,没有域名可随便填写
# 动态交给tomcat集群处理
location / {
proxy_pass http://tomcats;
proxy_redirect default;
}
# 静态处理交给nginx
location ~ .*\.(htm|html|gif|jpg|jpeg|png|bmp|swf|ioc|ico|rar|zip|txt|flv|mid|doc|ppt|pdf|xls|mp3|wma|css|js)$ {
root /usr/share/nginx/html/dist;
index index.html;
}
error_page 500 502 503 504 /50x.html;
location = /50x.html {
root html;
}
}
}
- 创建nginx容器 #
docker run
–name mynginx02
-d -p 80:80
-v /zking/nginx/conf/nginx.conf:/etc/nginx/nginx.conf
-v /zking/nginx/log:/var/log/nginx
-v /zking/nginx/html:/usr/share/nginx/html
nginx:latest
-
上传dist.zip的包到宿主机/zking/nginx/html目录下
rz
-
解压dist.zip
unzip dist.zip
-
访问vue项目首页
http://192.168.148.129/index.html注1:如果容器创建失败,可通过如下命令,查看在docker容器启动日志
docker logs -f -t --tail 行数 容器名
docker logs -f -t --tail 100 mynginx02注2:进入容器内部(/bin/bash 还是/bin/sh取决于跟镜像是centos还是alpine)
docker exec -it mynginx02 /bin/sh
java后台代码(oa项目)
第一步:创建一个名为oa的项目
第二步:配置pom.xml
pom.xml
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.2.2.RELEASE</version>
<relativePath/> <!-- lookup parent from repository -->
</parent>
<groupId>com.zking</groupId>
<artifactId>oa</artifactId>
<version>0.0.1-SNAPSHOT</version>
<name>oa</name>
<packaging>jar</packaging>
<description>Demo project for Spring Boot</description>
<properties>
<java.version>1.8</java.version>
</properties>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<optional>true</optional>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
<exclusions>
<exclusion>
<groupId>org.junit.vintage</groupId>
<artifactId>junit-vintage-engine</artifactId>
</exclusion>
</exclusions>
</dependency>
</dependencies>
<build>
<plugins>
<!--添加maven插件-->
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<!--添加自己的启动类路径!-->
<mainClass>com.zking.oa.OaApplication</mainClass>
</configuration>
<executions>
<execution>
<goals>
<!--可以把依赖的包都打包到生成的Jar包中-->
<goal>repackage</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
</build>
</project>
第三步:配置application.yml
application.yml
server:
servlet:
context-path: /oa
port: 8080
第四步:在 src/main/java/com/zking/oa 中创建 config 和 controller 两个文件夹,在文件夹中分别创建 CorsConfiguration.java 和 LoginController.java 两个java文件
CorsConfiguration.java
package com.zking.oa.config;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.CorsRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
@Configuration
public class CorsConfiguration {
/**
* 页面跨域访问Controller过滤
*
* @return
*/
@Bean
public WebMvcConfigurer corsConfigurer() {
return new WebMvcConfigurer() {
@Override
public void addCorsMappings(CorsRegistry registry) {
registry.addMapping("/*")//设置允许跨域的路径
.allowedOrigins("*") //设置允许跨域请求的域名
.allowedHeaders("*") // 设置允许跨域请求的请求头,例如:jwt
.allowCredentials(true)//是否允许证书 不再默认开启,允许跨域带上cookies
.allowedMethods("GET", "POST", "DELETE", "PUT", "PATCH")//设置允许的方法
.maxAge(3600);//跨域允许时间
}
};
}
}
LoginController.java
package com.zking.oa.controller;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
import java.net.InetAddress;
import java.util.HashMap;
import java.util.Map;
@RestController
public class LoginController {
@RequestMapping(value = "/login")
public Map<String, Object> login(String username, String password) throws Exception {
String ip = InetAddress.getLocalHost().getHostAddress();
Map<String, Object> map = new HashMap<String, Object>();
map.put("code", 0);
map.put("message", "用户" + username + "登陆成功");
map.put("ip", ip);//项目最终部署的服务器的ip地址
map.put("ts", System.currentTimeMillis() + "");
if (!"888888".equals(password)) {
map.put("code", -1);
map.put("message", "用户" + username + "登陆失败");
}
return map;
}
}
第五步:修改在 src/main/java/com/zking/oa 中的 OaApplication.java 文件
OaApplication.java
package com.zking.oa;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class OaApplication {
public static void main(String[] args) {
SpringApplication.run(OaApplication.class, args);
}
}
html前端代码
第一步:创建一个 spa(vue) 项目
第二步:把 config 文件夹中 index.js 中的端口号改为80
第三步:在 src 文件夹中添加 api 文件夹,跟 api 文件夹中的 action.js 和 http.js
action.js
/**
* 对后台请求的地址的封装,URL格式如下:
* 模块名_实体名_操作
*/
export default {
//nginx服务器ip地址:80端口
'SERVER': 'http://www.zking.com/api/crm',
// 'SERVER': 'http://localhost:8080/oa',
//用户管理
'SYS_USER_DOLOGIN': '/login', //登陆
//获得请求的完整地址,用于mockjs测试时使用
'getFullPath': k => {
return this.SERVER + this[k];
}
}
http.js
/**
* vue项目对axios的全局配置
*/
import axios from 'axios'
import qs from 'qs'
//引入action模块,并添加至axios的类属性urls上
import action from '@/api/action'
axios.urls = action
// axios默认配置
axios.defaults.timeout = 10000; // 超时时间
// axios.defaults.baseURL = 'http://localhost:8080/j2ee15'; // 默认地址
axios.defaults.baseURL = action.SERVER;
//整理数据
// 只适用于 POST,PUT,PATCH,transformRequest` 允许在向服务器发送前,修改请求数据
axios.defaults.transformRequest = function(data) {
data = qs.stringify(data);
return data;
};
// 请求拦截器
axios.interceptors.request.use(function(config) {
return config;
}, function(error) {
return Promise.reject(error);
});
// 响应拦截器
axios.interceptors.response.use(function(response) {
return response;
}, function(error) {
return Promise.reject(error);
});
// // 路由请求拦截
// // http request 拦截器
// axios.interceptors.request.use(
// config => {
// //config.data = JSON.stringify(config.data);
// //config.headers['Content-Type'] = 'application/json;charset=UTF-8';
// //config.headers['Token'] = 'abcxyz';
// //判断是否存在ticket,如果存在的话,则每个http header都加上ticket
// // if (cookie.get("token")) {
// // //用户每次操作,都将cookie设置成2小时
// // cookie.set("token", cookie.get("token"), 1 / 12)
// // cookie.set("name", cookie.get("name"), 1 / 12)
// // config.headers.token = cookie.get("token");
// // config.headers.name = cookie.get("name");
// // }
// return config;
// },
// error => {
// return Promise.reject(error.response);
// });
// // 路由响应拦截
// // http response 拦截器
// axios.interceptors.response.use(
// response => {
// if (response.data.resultCode == "404") {
// console.log("response.data.resultCode是404")
// // 返回 错误代码-1 清除ticket信息并跳转到登录页面
// // cookie.del("ticket")
// // window.location.href='http://login.com'
// return
// } else {
// return response;
// }
// },
// error => {
// return Promise.reject(error.response) // 返回接口返回的错误信息
// });
export default axios;
第四步:把 src 中的 App.vue 和 main.js 两个文件进行修改
App.vue
<template>
<div id="app">
<!-- <img src="./assets/logo.png"> -->
<router-view/>
</div>
</template>
<script>
export default {
name: 'App'
}
</script>
<style>
#app {
font-family: 'Avenir', Helvetica, Arial, sans-serif;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
text-align: center;
color: #2c3e50;
margin-top: 60px;
}
</style>
main.js
// The Vue build version to load with the `import` command
// (runtime-only or standalone) has been set in webpack.base.conf with an alias.
import Vue from 'vue'
import ElementUI from 'element-ui' //新添加1
import 'element-ui/lib/theme-chalk/index.css' //新添加2,避免后期打包样式不同,要放在import App from './App';之前
import App from './App'
import router from './router'
//引入axios
import axios from '@/api/http'
import VueAxios from 'vue-axios'
//vue加载插件
Vue.use(ElementUI)
Vue.use(VueAxios, axios)
Vue.config.productionTip = false
/* eslint-disable no-new */
new Vue({
el: '#app',
router,
components: { App },
template: '<App/>'
})
第五步:在 src 中添加 views 文件夹 ,在 views 中添加 Login.vue 登录页面
Login.vue
<template>
<div class="login-wrap">
<el-form class="login-container" :model="loginForm" :rules="loginFormRules" ref="loginForm">
<h3 class="title">用户登陆</h3>
<el-form-item label="" prop="username">
<el-input v-model="loginForm.username" auto-complete="off" placeholder="帐号"></el-input>
</el-form-item>
<el-form-item label="" prop="password">
<el-input v-model="loginForm.password" auto-complete="off" placeholder="密码" show-password></el-input>
</el-form-item>
<el-form-item label="">
<el-button type="primary" style="width:100%;" v-on:click="doLogin()">登陆</el-button>
</el-form-item>
<el-row style="text-align: center;">
<el-link type="primary" href="#">忘记密码</el-link>
<el-link type="primary" href="#">快速注册</el-link>
</el-row>
</el-form>
</div>
</template>
<script>
//import action from '@/api/action.js'
export default {
data: function() {
return {
'base64': null,
loginForm: {
username: null,
password: null
},
loginFormRules: {
username: [{
required: true,
message: '请输入帐号',
trigger: 'blur'
},
{
min: 1,
max: 10,
message: '长度在 1 到 10 个字符',
trigger: 'blur'
}
],
password: [{
required: true,
message: '请输入密码',
trigger: 'blur'
},
{
min: 1,
max: 10,
message: '长度在 1 到 10 个字符',
trigger: 'blur'
}
]
}
}
},
methods: {
doGetBase64: function() {
console.log('doGetBase64');
let url = this.axios.urls.SYS_CAPTCHA_CODE;
this.axios.get(url).then((response) => {
this.base64 = response.data.base64;
this.loginForm.jwt = response.data.jwt;
//console.log(this.base64);
// this.src = "data:image/png;base64," + this.base64;
}).catch((response) => {
console.log(response);
});
},
toReg: function() {
console.log('toReg');
this.$router.push('/Reg');
},
doLogin: function() {
console.log("doLogin");
this.$refs['loginForm'].validate((valid) => {
if (false === valid) {
return false;
}
console.log('向服务器提交数据');
let url = this.axios.urls.SYS_USER_DOLOGIN;
this.axios.post(url, this.loginForm).then((response) => {
if (0 == response.data.code) {
this.$message({
message: response.data.message,
type: 'success'
});
this.$store.commit('setUsername', {
username: this.username
});
this.$router.push('/AppMain');
} else {
this.$message.error(response.data.message);
setTimeout(() => { //延时0.3秒刷新验证码
this.doGetBase64();
}, 300);
}
}).catch((response) => {
console.log(response);
});
});
}
},
created: function() {
// console.log('created');
// setTimeout(() => {
// this.doGetBase64();
// }, 300);
this.$nextTick(function() {
this.doGetBase64();
})
},
// mounted: function() {
// this.$nextTick(function() {
// this.doGetBase64();
// })
// },
computed: {
src: function() {
return "data:image/png;base64," + this.base64;
}
}
};
</script>
<style scoped>
.login-wrap {
box-sizing: border-box;
width: 100%;
height: 100%;
padding-top: 10%;
background-image: url();
/* background-color: #112346; */
background-repeat: no-repeat;
background-position: center right;
background-size: 100%;
}
.login-container {
border-radius: 10px;
margin: 0px auto;
width: 350px;
padding: 30px 35px 15px 35px;
background: #fff;
border: 1px solid #eaeaea;
text-align: left;
box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);
}
.title {
margin: 0px auto 40px auto;
text-align: center;
color: #505458;
}
.code-box {
text-align: right;
}
.codeimg {
cursor: pointer;
height: 40px;
}
</style>
第六步:在 src/router/index.js 下添加代码
index.js
import Vue from 'vue'
import Router from 'vue-router'
import HelloWorld from '@/components/HelloWorld'
//登陆
import Login from '@/views/Login'
Vue.use(Router)
export default new Router({
routes: [
{
path: '/',
name: 'Login',
component: Login
}
]
})
第七步:在 src/components 下添加 HelloWorld.vue
HelloWorld.vue
<template>
<div class="hello">
<h1>{{ msg }}</h1>
<h2>Essential Links</h2>
<ul>
<li>
<a
href="https://vuejs.org"
target="_blank"
>
Core Docs
</a>
</li>
<li>
<a
href="https://forum.vuejs.org"
target="_blank"
>
Forum
</a>
</li>
<li>
<a
href="https://chat.vuejs.org"
target="_blank"
>
Community Chat
</a>
</li>
<li>
<a
href="https://twitter.com/vuejs"
target="_blank"
>
Twitter
</a>
</li>
<br>
<li>
<a
href="http://vuejs-templates.github.io/webpack/"
target="_blank"
>
Docs for This Template
</a>
</li>
</ul>
<h2>Ecosystem</h2>
<ul>
<li>
<a
href="http://router.vuejs.org/"
target="_blank"
>
vue-router
</a>
</li>
<li>
<a
href="http://vuex.vuejs.org/"
target="_blank"
>
vuex
</a>
</li>
<li>
<a
href="http://vue-loader.vuejs.org/"
target="_blank"
>
vue-loader
</a>
</li>
<li>
<a
href="https://github.com/vuejs/awesome-vue"
target="_blank"
>
awesome-vue
</a>
</li>
</ul>
</div>
</template>
<script>
export default {
name: 'HelloWorld',
data () {
return {
msg: 'Welcome to Your Vue.js App'
}
}
}
</script>
<!-- Add "scoped" attribute to limit CSS to this component only -->
<style scoped>
h1, h2 {
font-weight: normal;
}
ul {
list-style-type: none;
padding: 0;
}
li {
display: inline-block;
margin: 0 10px;
}
a {
color: #42b983;
}
</style>
附录
- 附录一:docker 镜像下安装Vim
- 进入docker镜像
docker exec -it 镜像Id或镜像name /bin/bash
-
附录二: nginx.conf配置常用命令
vim /usr/local/nginx/conf/nginx.conf # 编辑主配置文件测试Nginx配置是否正确命令,先进入主程序的安装目录
cd /usr/local/nginx/conf
/usr/local/nginx/sbin/nginx -t # 判断Nginx配置是否正确命令
tail -100f /var/log/nginx/access.log # 查看Nginx访问日志,必须以nginx.conf配置文件的目录一致service nginx start # 启动
service nginx stop # 停止
service nginx restart # 重新启动
service nginx reload # 重新加载配置,平滑启动
service nginx status # 查看nginx状态ps -ef | grep nginx # 查看Nginx安装目录,master process后面的就是nginx的目录
cd /usr/sbin # 进入nginx主程序的安装目录
/usr/sbin/nginx -V # -v 显示 nginx 的版本,本示例安装版本为:nginx version: nginx/1.14.2 (Ubuntu)
/usr/sbin/nginx -V # -V 显示 nginx 的版本,编译器版本和配置参数 -
附录三:hbuilderX打包vue项目白屏问题
将项目目录下的config文件夹里的index.js文件中,将build对象下的assetsPublicPath中的“/”,改为“./”后,再打包生成的 dist 文件
/config/index.js
assetsPublicPath:"/" -
附录四:springboot中如何解决跨域访问
见:资料/CorsConfiguration.java