本次分享主要是针对,小公司及初创团队如何用较低成本落地微服务,拥抱变化,快速交付
 从 百度指数 搜索 微服务 、 spring boot 、 spring cloud 、 dubbo 相关关键词,得到如下趋势(微服务的概念源于2014年3月Martin Fowler所写的一篇文章 Microservices ,所以选择从2014.03至今) 
  
 
 从图中可见, dubbo 的搜索量增势放缓,而 Spring Boot 从16年中下旬开始发力,一路高涨。而学习了 Spring boot 再学习 Spring Cloud 几乎顺理成章。 
 spring boot旨在解决Spring越来越臃肿的全家桶方案的 配置地狱 (讽刺的是,Spring刚出道是扯着轻量化解决方案大旗一路冲杀,现在自己也开始慢慢胖起来了),提供了很多简单易用的 starter 。特点是预定大于配置。 
dubbo放缓是源于,阿里巴巴中间断更将近三年( dubbo-2.4.11 2014-10-30, dubbo-2.5.4 2017-09-07),很多依赖框架和技术都较为陈旧,也不接纳社区的PR(当然,从17年九月份开始恢复更新,后面会有说到),导致当当另起炉灶,fork了一个 dangdangdotcom/dubbox 当然,现在也已断更。而且dubbo仅相当于Spring cloud的一个子集,参考 微服务架构的基础框架选择:Spring Cloud还是Dubbo? (此处说的是dubbo2.x,最新的3.x变化较大,后边会说到)
 k8s 、 kubernetes 、 docker 的搜索趋势 
  
 
下面是我整理的一些关于单体服务和微服务的对比
单体应用好处
单体应用缺点
微服务好处
微服务的缺点
扩展阅读 Introduction to Microservices
下面讲一下,我司在落地微服务时的框架选型方面的一些经验。
 公司主要使用java,所以决定使用spring 框架中的 spring cloud 作为微服务基础框架,但是原生 spring cloud 学习曲线比较陡峭,需要学习 feign , zuul , eureka , hystrix , zipkin , ribbon … 所谓的Spring Cloud全家桶 
综合考虑团队的技术水平和学习成本,最后采用了国外的开源框架 jhipster官网 jhipster github ,登记在册的,使用jhipster的企业有224家(截止2018-01-11),包括埃森哲,google,adobe等大厂。
jhipster 是由2013年由法国Java专家 Julien Dubois (朱利安 杜波尔斯)率先倡导,至今已有快5年了,积累了大量丰富经验。 采用Java 8(目前尚不支持java9,但是有开发计划),特色是多用注解, 不用XML 配置的组态,配备了全方位的工作环境,从开发,测试,监控到制成,以及云部署。
国内用dubbo的较多,用jhipster的较少,起码很多群里交流的时候,很多表示没听过,或者是我加的假群?至于为啥不用dubbo,前面提到过,一个是中间断更,以及阿里说不更就不更的优良传统,还有dubbo从功能来说,只是Spring Cloud的一个子集(dubbo 2.x) 。
列举一下 jhipster 给的技术栈 ,参见 Technology stack
dev , prod ) 其实真正用过就会发现,jhipster支持的不止列表中描述的这些。
如果不会用yarn或者不方便用命令行生成项目,可以使用 jhipster online
如果想学习jhipster,可以参考我在公司推广jhipster时写的一本gitbook jhipster开发笔记
 同时,值得一提的是,jhipster也支持通过 jhipster rancher-compose 命令来生成 rancher-compose.yml 和 docker-compose.yml 参见 [BETA] Deploying to Rancher 
对于小团队落地微服务,可以考虑使用jhipster来生成项目,能够极大的提高效率。基本上可以视作jhipster是一套基于spring boot的最佳实践(不仅支持微服务,也支持单体式应用)。
对于想学习spring boot或者spring cloud的也建议了解一下jhipster,好过独自摸索
jhipster依赖的技术框架版本基本都是最新稳定版,版本更新比较及时,基本上一月一个版本,对github上的issues和pr响应比较及时(一般在24小时内)
如果是windows nodejs 需要安装v7.x,因为注册中心和网关需要用到 node-sass@4.5.0 ,但是github上的node-sass的rebuild只有v7.x(process 51) 版本的,而自己构建太反人类了。如果是linux,可以尝试高版本的。
安装完 node,yarn后,执行下面代码,使用npm的淘宝镜像,加速构建。
yarn config set sass_binary_site https://npm.taobao.org/mirrors/node-sass/ yarn config set phantomjs_cdnurl https://npm.taobao.org/mirrors/phantomjs/ yarn config set registry https://registry.npm.taobao.org
安装 jdk8,maven,maven加速,请自行百度。
下载并运行 注册中心-jhipster-registry 详细文档,参见 The JHipster Registry
$ git clone https://github.com/jhipster/jhipster-registry.git
$ cd jhipster-registry
$ yarn install
$ mvnw
##....
----------------------------------------------------------
        Application 'jhipster-registry' is running! Access URLs:
        Local:          http://localhost:8761
        External:       http://xx.xx.xx.xx:8761
        Profile(s):     [swagger, dev, native]
----------------------------------------------------------
 
   浏览器访问 http://localhost:8761 初始用户名密码均为 admin 
  
 
spring config server,统一配置中心,可以统一管理不同环境的数据库地址,用户名,密码等敏感数据
  
 
jhipster registry 对应SC(Spring Cloud)的eurake+spring config server,想想自己用原生的SC自己搞的辛酸泪吧,再牛逼,刚学,也得10min+才能跑起来吧?
创建api网关,参见 Creating an application 和 The JHipster API Gateway
$ yarn global add generator-jhipster
$ mkdir gateway
$ cd gateway
$ yo jhipster
        ██╗ ██╗   ██╗ ████████╗ ███████╗   ██████╗ ████████╗ ████████╗ ███████╗
        ██║ ██║   ██║ ╚══██╔══╝ ██╔═══██╗ ██╔════╝ ╚══██╔══╝ ██╔═════╝ ██╔═══██╗
        ██║ ████████║    ██║    ███████╔╝ ╚█████╗     ██║    ██████╗   ███████╔╝
  ██╗   ██║ ██╔═══██║    ██║    ██╔════╝   ╚═══██╗    ██║    ██╔═══╝   ██╔══██║
  ╚██████╔╝ ██║   ██║ ████████╗ ██║       ██████╔╝    ██║    ████████╗ ██║  ╚██╗
   ╚═════╝  ╚═╝   ╚═╝ ╚═══════╝ ╚═╝       ╚═════╝     ╚═╝    ╚═══════╝ ╚═╝   ╚═╝
                            http://www.jhipster.tech
Welcome to the JHipster Generator v4.13.2
 _______________________________________________________________________________________________________________
  If you find JHipster useful consider supporting our collective https://opencollective.com/generator-jhipster
  Documentation for creating an application: http://www.jhipster.tech/creating-an-app/
 _______________________________________________________________________________________________________________
Application files will be generated in folder: G:/jh/gateway
? Which *type* of application would you like to create? Microservice gateway
? What is the base name of your application? gateway
? As you are running in a microservice architecture, on which port would like your server to run? It should be unique to avoid port conflicts. 8080
? What is your default Java package name? com.anjia.gateway
? Which service discovery server do you want to use? JHipster Registry (uses Eureka, provides Spring Cloud Config support and monitoring dashboards)
? Which *type* of authentication would you like to use? JWT authentication (stateless, with a token)
? Which *type* of database would you like to use? SQL (H2, MySQL, MariaDB, PostgreSQL, Oracle, MSSQL)
? Which *production* database would you like to use? MySQL
? Which *development* database would you like to use? H2 with disk-based persistence
? Do you want to use Hibernate 2nd level cache? Yes
? Would you like to use Maven or Gradle for building the backend? Maven
? Which other technologies would you like to use?
? Which *Framework* would you like to use for the client? Angular 5
? Would you like to enable *SASS* support using the LibSass stylesheet preprocessor? No
? Would you like to enable internationalization support? Yes
? Please choose the native language of the application English
? Please choose additional languages to install
? Besides JUnit and Karma, which testing frameworks would you like to use?
? Would you like to install other generators from the JHipster Marketplace? No
$ mvnw
----------------------------------------------------------
        Application 'gateway' is running! Access URLs:
        Local:          http://localhost:8080
        External:       http://xx.xx.xx.xx:8080
        Profile(s):     [swagger, dev]
----------------------------------------------------------
 
    
 
 访问 http://localhost:8080/ 默认用户名密码均为 admin 
  
 
创建服务 参考 Creating an application
$ mkdir foo
$ cd foo
$ yo jhipster
        ██╗ ██╗   ██╗ ████████╗ ███████╗   ██████╗ ████████╗ ████████╗ ███████╗
        ██║ ██║   ██║ ╚══██╔══╝ ██╔═══██╗ ██╔════╝ ╚══██╔══╝ ██╔═════╝ ██╔═══██╗
        ██║ ████████║    ██║    ███████╔╝ ╚█████╗     ██║    ██████╗   ███████╔╝
  ██╗   ██║ ██╔═══██║    ██║    ██╔════╝   ╚═══██╗    ██║    ██╔═══╝   ██╔══██║
  ╚██████╔╝ ██║   ██║ ████████╗ ██║       ██████╔╝    ██║    ████████╗ ██║  ╚██╗
   ╚═════╝  ╚═╝   ╚═╝ ╚═══════╝ ╚═╝       ╚═════╝     ╚═╝    ╚═══════╝ ╚═╝   ╚═╝
                            http://www.jhipster.tech
Welcome to the JHipster Generator v4.13.2
 _______________________________________________________________________________________________________________
  If you find JHipster useful consider supporting our collective https://opencollective.com/generator-jhipster
  Documentation for creating an application: http://www.jhipster.tech/creating-an-app/
 _______________________________________________________________________________________________________________
Application files will be generated in folder: G:/jh/foo
? Which *type* of application would you like to create? Microservice application
? What is the base name of your application? foo
? As you are running in a microservice architecture, on which port would like your server to run? It should be unique to avoid port conflicts. 8081
? What is your default Java package name? com.anjia.foo
? Which service discovery server do you want to use? JHipster Registry (uses Eureka, provides Spring Cloud Config support and monitoring dashboards)
? Which *type* of authentication would you like to use? JWT authentication (stateless, with a token)
? Which *type* of database would you like to use? SQL (H2, MySQL, MariaDB, PostgreSQL, Oracle, MSSQL)
? Which *production* database would you like to use? MySQL
? Which *development* database would you like to use? H2 with disk-based persistence
? Do you want to use the Spring cache abstraction? Yes, with the Hazelcast implementation (distributed cache, for multiple nodes)
? Do you want to use Hibernate 2nd level cache? Yes
? Would you like to use Maven or Gradle for building the backend? Maven
? Which other technologies would you like to use?
? Would you like to enable internationalization support? Yes
? Please choose the native language of the application Chinese (Simplified)
? Please choose additional languages to install
? Besides JUnit and Karma, which testing frameworks would you like to use?
? Would you like to install other generators from the JHipster Marketplace? No
$ mvnw
----------------------------------------------------------
        Application 'foo' is running! Access URLs:
        Local:          http://localhost:8081
        External:       http://xx.xx.xx.xx:8081
        Profile(s):     [swagger, dev]
----------------------------------------------------------
 
   访问 http://localhost:8080/#/docs 默认用户名密码均为 admin ,使用 swagger 管理api文档,开发时,仅需要添加对应的注解,即可自动生成文档,解决了传统通过word,pdf等管理接口时,文档更新不及时等问题。并且可以通过 try it 可以直接调用接口,避免了接口调试时使用 curl , postman 等工具 
  
 
至此,已经创建了一个简单微服务(jhipster-registry是注册中心,gateway是网关,foo是具体的功能模块)。
参考文档 Creating an entity
 jhipster支持通过命令行创建实体,也支持uml或jdl生成实体,为了省事,此处使用 官方 jdl-studio 的默认jdl文件 https://start.jhipster.tech/jdl-studio/ 
  
 
$ yo jhipster:import-jdl /path/to/jdl-studio/jhipster-jdl.jh
The jdl is being parsed.
Writing entity JSON files.
Updated entities are: Region,Country,Location,Department,Task,Employee,Job,JobHistory
Generating entities.
Found the .jhipster/Region.json configuration file, entity can be automatically generated!
The entity Region is being updated.
Found the .jhipster/Country.json configuration file, entity can be automatically generated!
The entity Country is being updated.
Found the .jhipster/Location.json configuration file, entity can be automatically generated!
The entity Location is being updated.
Found the .jhipster/Department.json configuration file, entity can be automatically generated!
The entity Department is being updated.
Found the .jhipster/Task.json configuration file, entity can be automatically generated!
The entity Task is being updated.
Found the .jhipster/Employee.json configuration file, entity can be automatically generated!
The entity Employee is being updated.
Found the .jhipster/Job.json configuration file, entity can be automatically generated!
The entity Job is being updated.
Found the .jhipster/JobHistory.json configuration file, entity can be automatically generated!
The entity JobHistory is being updated.
   create src/main/resources/config/liquibase/changelog/20180107064934_added_entity_Region.xml
   create src/main/java/com/anjia/foo/domain/Region.java
   create src/main/java/com/anjia/foo/repository/RegionRepository.java
   create src/main/java/com/anjia/foo/web/rest/RegionResource.java
   create src/main/java/com/anjia/foo/service/RegionService.java
   create src/main/java/com/anjia/foo/service/impl/RegionServiceImpl.java
   create src/main/java/com/anjia/foo/service/dto/RegionDTO.java
   create src/main/java/com/anjia/foo/service/mapper/EntityMapper.java
   create src/main/java/com/anjia/foo/service/mapper/RegionMapper.java
   create src/test/java/com/anjia/foo/web/rest/RegionResourceIntTest.java
 conflict src/main/resources/config/liquibase/master.xml
? Overwrite src/main/resources/config/liquibase/master.xml? overwrite
    force src/main/resources/config/liquibase/master.xml
   create src/main/resources/config/liquibase/changelog/20180107064935_added_entity_Country.xml
   create src/main/resources/config/liquibase/changelog/20180107064935_added_entity_constraints_Country.xml
   create src/main/java/com/anjia/foo/domain/Country.java
   create src/main/java/com/anjia/foo/repository/CountryRepository.java
   create src/main/java/com/anjia/foo/web/rest/CountryResource.java
   create src/main/java/com/anjia/foo/service/CountryService.java
   create src/main/java/com/anjia/foo/service/impl/CountryServiceImpl.java
   create src/main/java/com/anjia/foo/service/dto/CountryDTO.java
   create src/main/java/com/anjia/foo/service/mapper/CountryMapper.java
   create src/test/java/com/anjia/foo/web/rest/CountryResourceIntTest.java
   create src/main/resources/config/liquibase/changelog/20180107064936_added_entity_Location.xml
   create src/main/resources/config/liquibase/changelog/20180107064936_added_entity_constraints_Location.xml
   create src/main/java/com/anjia/foo/domain/Location.java
   create src/main/java/com/anjia/foo/repository/LocationRepository.java
   create src/main/java/com/anjia/foo/web/rest/LocationResource.java
   create src/main/java/com/anjia/foo/service/LocationService.java
   create src/main/java/com/anjia/foo/service/impl/LocationServiceImpl.java
   create src/main/java/com/anjia/foo/service/dto/LocationDTO.java
   create src/main/java/com/anjia/foo/service/mapper/LocationMapper.java
   create src/test/java/com/anjia/foo/web/rest/LocationResourceIntTest.java
   create src/main/resources/config/liquibase/changelog/20180107064937_added_entity_Department.xml
   create src/main/resources/config/liquibase/changelog/20180107064937_added_entity_constraints_Department.xml
   create src/main/java/com/anjia/foo/domain/Department.java
   create src/main/java/com/anjia/foo/repository/DepartmentRepository.java
   create src/main/java/com/anjia/foo/web/rest/DepartmentResource.java
   create src/main/java/com/anjia/foo/service/DepartmentService.java
   create src/main/java/com/anjia/foo/service/impl/DepartmentServiceImpl.java
   create src/main/java/com/anjia/foo/service/dto/DepartmentDTO.java
   create src/main/java/com/anjia/foo/service/mapper/DepartmentMapper.java
   create src/test/java/com/anjia/foo/web/rest/DepartmentResourceIntTest.java
   create src/main/resources/config/liquibase/changelog/20180107064938_added_entity_Task.xml
   create src/main/java/com/anjia/foo/domain/Task.java
   create src/main/java/com/anjia/foo/repository/TaskRepository.java
   create src/main/java/com/anjia/foo/web/rest/TaskResource.java
   create src/main/java/com/anjia/foo/service/TaskService.java
   create src/main/java/com/anjia/foo/service/impl/TaskServiceImpl.java
   create src/main/java/com/anjia/foo/service/dto/TaskDTO.java
   create src/main/java/com/anjia/foo/service/mapper/TaskMapper.java
   create src/test/java/com/anjia/foo/web/rest/TaskResourceIntTest.java
   create src/main/resources/config/liquibase/changelog/20180107064939_added_entity_Employee.xml
   create src/main/resources/config/liquibase/changelog/20180107064939_added_entity_constraints_Employee.xml
   create src/main/java/com/anjia/foo/domain/Employee.java
   create src/main/java/com/anjia/foo/repository/EmployeeRepository.java
   create src/main/java/com/anjia/foo/web/rest/EmployeeResource.java
   create src/main/java/com/anjia/foo/service/dto/EmployeeDTO.java
   create src/main/java/com/anjia/foo/service/mapper/EmployeeMapper.java
   create src/test/java/com/anjia/foo/web/rest/EmployeeResourceIntTest.java
   create src/main/resources/config/liquibase/changelog/20180107064940_added_entity_Job.xml
   create src/main/resources/config/liquibase/changelog/20180107064940_added_entity_constraints_Job.xml
   create src/main/java/com/anjia/foo/domain/Job.java
   create src/main/java/com/anjia/foo/repository/JobRepository.java
   create src/main/java/com/anjia/foo/web/rest/JobResource.java
   create src/main/java/com/anjia/foo/service/dto/JobDTO.java
   create src/main/java/com/anjia/foo/service/mapper/JobMapper.java
   create src/test/java/com/anjia/foo/web/rest/JobResourceIntTest.java
   create src/main/resources/config/liquibase/changelog/20180107064941_added_entity_JobHistory.xml
   create src/main/resources/config/liquibase/changelog/20180107064941_added_entity_constraints_JobHistory.xml
   create src/main/java/com/anjia/foo/domain/JobHistory.java
   create src/main/java/com/anjia/foo/repository/JobHistoryRepository.java
   create src/main/java/com/anjia/foo/web/rest/JobHistoryResource.java
   create src/main/java/com/anjia/foo/service/JobHistoryService.java
   create src/main/java/com/anjia/foo/service/impl/JobHistoryServiceImpl.java
   create src/main/java/com/anjia/foo/service/dto/JobHistoryDTO.java
   create src/main/java/com/anjia/foo/service/mapper/JobHistoryMapper.java
   create src/test/java/com/anjia/foo/web/rest/JobHistoryResourceIntTest.java
   create src/main/java/com/anjia/foo/domain/enumeration/Language.java
 
   重启 foo 服务,再次访问 http://localhost:8080/#/docs 发现多了很多接口 
  
 
 通过swagger ui,找到 region-resource 找到 POST /api/regions 创建一个名为 test 的 regison 
  
 
 点 try it out! 然后浏览器打开 h2 数据库 http://localhost:8081/h2-console 
  
  
 
 查询 REGION 表,数据已经插入成功。 
至此,一个虽然简单,但是可用的微服务已经弄好。
参见文档 [BETA] Deploying to Rancher ,jhipster支持发布到 Cloud Foundry , Heroku , Kubernetes , Openshift , Rancher , AWS , Boxfuse
建议使用rancher,原因, Cloud Foundry , Heroku , AWS , Boxfuse 都是云环境,k8s和openshift origin太复杂了,而rancher很容易上手,其联合创始人成为CNCF的理事会成员。
附上一张 CNCF天梯图
  
 
 $ mkdir docker
 $ cd docker
 $ yo jhipster rancher-rancher
        ██╗ ██╗   ██╗ ████████╗ ███████╗   ██████╗ ████████╗ ████████╗ ███████╗
        ██║ ██║   ██║ ╚══██╔══╝ ██╔═══██╗ ██╔════╝ ╚══██╔══╝ ██╔═════╝ ██╔═══██╗
        ██║ ████████║    ██║    ███████╔╝ ╚█████╗     ██║    ██████╗   ███████╔╝
  ██╗   ██║ ██╔═══██║    ██║    ██╔════╝   ╚═══██╗    ██║    ██╔═══╝   ██╔══██║
  ╚██████╔╝ ██║   ██║ ████████╗ ██║       ██████╔╝    ██║    ████████╗ ██║  ╚██╗
   ╚═════╝  ╚═╝   ╚═╝ ╚═══════╝ ╚═╝       ╚═════╝     ╚═╝    ╚═══════╝ ╚═╝   ╚═╝
                            http://www.jhipster.tech
Welcome to the JHipster Generator v4.13.2
 _______________________________________________________________________________________________________________
  If you find JHipster useful consider supporting our collective https://opencollective.com/generator-jhipster
  Documentation for creating an application: http://www.jhipster.tech/creating-an-app/
 _______________________________________________________________________________________________________________
Application files will be generated in folder: G:/jh/docker
? What is the base name of your application? (docker)
G:/jh/docker
λ  jhipster rancher-compose
Using JHipster version installed globally
Executing jhipster:rancher-compose
Options:
:cow: [BETA] Welcome to the JHipster Rancher Compose Generator :cow:
Files will be generated in folder: G:/jh/docker
? Which *type* of application would you like to deploy? Microservice application
? Enter the root directory where your gateway(s) and microservices are located ../
2 applications found at G:/jh/
? Which applications do you want to include in your configuration? foo, gateway
? Do you want to setup monitoring for your applications ? Yes, for logs and metrics with the JHipster Console (based on ELK and Zipkin)
JHipster registry detected as the service discovery and configuration provider used by your apps
? Enter the admin password used to secure the JHipster Registry admin
? Would you like to enable rancher load balancing support? Yes
? What should we use for the base Docker repository name?
? What command should we use for push Docker image to repository? docker push
Checking Docker images in applications' directories...
ls: no such file or directory: G:/jh/foo/target/docker
ls: no such file or directory: G:/jh/gateway/target/docker
   create rancher-compose.yml
   create docker-compose.yml
   create registry-config-sidekick/Dockerfile
   create registry-config-sidekick/application.yml
Rancher Compose configuration generated with missing images!
To generate the missing Docker image(s), please run:
  ./mvnw verify -Pprod dockerfile:build in G:/jh/foo
  ./mvnw verify -Pprod dockerfile:build in G:/jh/gateway
WARNING! You will need to push your image to a registry. If you have not done so, use the following commands to tag and push the images:
docker push foo
docker push gateway
Congratulations, JHipster execution is complete!
 
  version: '2'
services:
    lb:
        # load balancer container
        scale: 1
        load_balancer_config:
          name: lb config
        health_check:
          port: 42
          interval: 2000
          unhealthy_threshold: 3
          healthy_threshold: 2
          response_timeout: 2000
    foo-app:
        scale: 1
    foo-mysql:
        scale: 1
    
    gateway-app:
        scale: 1
    gateway-mysql:
        scale: 1
    
    jhipster-registry:
        scale: 1
    jhipster-elasticsearch:
        scale: 1
    jhipster-logstash:
        scale: 1
    jhipster-console:
        scale: 1
    # Uncomment this section to enable Zipkin
    #jhipster-zipkin:
    #    scale: 1
 
  version: '2'
services:
    lb:
      image: rancher/load-balancer-service
      ports:
        # Listen on public port 80 and direct traffic to private port 8080 of the service
        - 80:8080
      links:
        # Target services in the same stack will be listed as a link
        - gateway-app:gateway-app
    foo-app:
        image: foo
        environment:
            - SPRING_PROFILES_ACTIVE=prod,swagger
            - EUREKA_CLIENT_SERVICE_URL_DEFAULTZONE=http://admin:$${jhipster.registry.password}@jhipster-registry:8761/eureka
            - SPRING_CLOUD_CONFIG_URI=http://admin:$${jhipster.registry.password}@jhipster-registry:8761/config
            - SPRING_DATASOURCE_URL=jdbc:mysql://foo-mysql:3306/foo?useUnicode=true&characterEncoding=utf8&useSSL=false
            - JHIPSTER_SLEEP=30
            - JHIPSTER_LOGGING_LOGSTASH_ENABLED=true
            - JHIPSTER_LOGGING_LOGSTASH_HOST=jhipster-logstash
            - JHIPSTER_METRICS_LOGS_ENABLED=true
            - JHIPSTER_METRICS_LOGS_REPORT_FREQUENCY=60
            - JHIPSTER_REGISTRY_PASSWORD=admin
    foo-mysql:
        image: mysql:5.7.20
        environment:
            - MYSQL_USER=root
            - MYSQL_ALLOW_EMPTY_PASSWORD=yes
            - MYSQL_DATABASE=foo
        command: >-
            mysqld --lower_case_table_names=1 --skip-ssl --character_set_server=utf8
            --explicit_defaults_for_timestamp
    
    gateway-app:
        image: gateway
        environment:
            - SPRING_PROFILES_ACTIVE=prod,swagger
            - EUREKA_CLIENT_SERVICE_URL_DEFAULTZONE=http://admin:$${jhipster.registry.password}@jhipster-registry:8761/eureka
            - SPRING_CLOUD_CONFIG_URI=http://admin:$${jhipster.registry.password}@jhipster-registry:8761/config
            - SPRING_DATASOURCE_URL=jdbc:mysql://gateway-mysql:3306/gateway?useUnicode=true&characterEncoding=utf8&useSSL=false
            - JHIPSTER_SLEEP=30
            - JHIPSTER_LOGGING_LOGSTASH_ENABLED=true
            - JHIPSTER_LOGGING_LOGSTASH_HOST=jhipster-logstash
            - JHIPSTER_METRICS_LOGS_ENABLED=true
            - JHIPSTER_METRICS_LOGS_REPORT_FREQUENCY=60
            - JHIPSTER_REGISTRY_PASSWORD=admin
        ports:
            - 8080:8080
    gateway-mysql:
        image: mysql:5.7.20
        environment:
            - MYSQL_USER=root
            - MYSQL_ALLOW_EMPTY_PASSWORD=yes
            - MYSQL_DATABASE=gateway
        command: >-
            mysqld --lower_case_table_names=1 --skip-ssl --character_set_server=utf8
            --explicit_defaults_for_timestamp
    
    jhipster-registry:
        image: jhipster/jhipster-registry:v3.2.4
        #volumes:
        #    - ./central-server-config:/central-config
        # By default the JHipster Registry runs with the "dev" and "native"
        # Spring profiles.
        # "native" profile means the filesystem is used to store data, see
        # http://cloud.spring.io/spring-cloud-config/spring-cloud-config.html
        environment:
            - SPRING_PROFILES_ACTIVE=dev,native
            - SECURITY_USER_PASSWORD=admin
            - JHIPSTER_LOGGING_LOGSTASH_ENABLED=true
            - JHIPSTER_LOGGING_LOGSTASH_HOST=jhipster-logstash
            - JHIPSTER_METRICS_LOGS_ENABLED=true
            - JHIPSTER_METRICS_LOGS_REPORTFREQUENCY=60
            - SPRING_CLOUD_CONFIG_SERVER_NATIVE_SEARCH_LOCATIONS=file:./config/
            # Uncomment to use a Git configuration source instead of the local filesystem
            # mounted from the registry-config-sidekick volume
            # - GIT_URI=https://github.com/jhipster/jhipster-registry/
            # - GIT_SEARCH_PATHS=central-config
        ports:
            - 8761:8761
        volumes:
          - /config
        volumes_from:
          - registry-config-sidekick
        labels:
          io.rancher.sidekicks: registry-config-sidekick
    registry-config-sidekick:
        # this docker image must be built with:
        # docker build -t registry-config-sidekick registry-config-sidekick
        image: registry-config-sidekick
        tty: true
        stdin_open: true
        command:
          - cat
        volumes:
          - config:/config
    jhipster-elasticsearch:
        image: jhipster/jhipster-elasticsearch:v2.2.1
        ports:
            - 9200:9200
            - 9300:9300
        # Uncomment this section to have elasticsearch data persisted to a volume
        #volumes:
        #   - ./log-data:/usr/share/elasticsearch/data
    jhipster-logstash:
        image: jhipster/jhipster-logstash:v2.2.1
        command: logstash -f /conf/logstash.conf
        ports:
            - 5000:5000/udp
        # Uncomment this section to have logstash config loaded from a volume
        #volumes:
        #    - ./log-conf/:/conf
    jhipster-console:
        image: jhipster/jhipster-console:v2.2.1
        ports:
            - 5601:5601
    # Uncomment this section to enable Zipkin
    #jhipster-zipkin:
    #    image: jhipster/jhipster-zipkin:v2.2.1
    #    ports:
    #        - 9411:9411
    #    environment:
    #        - ES_HOSTS=http://jhipster-elasticsearch:9200
    #        - ZIPKIN_UI_LOGS_URL=http://localhost:5601/app/kibana#/dashboard/logs-dashboard?_g=(refreshInterval:(display:Off,pause:!f,value:0),time:(from:now-1h,mode:quick,to:now))&_a=(filters:!(),options:(darkTheme:!f),panels:!((col:1,id:logs-levels,panelIndex:2,row:1,size_x:6,size_y:3,type:visualization),(col:7,columns:!(stack_trace),id:Stacktraces,panelIndex:7,row:1,size_x:4,size_y:3,sort:!('@timestamp',desc),type:search),(col:11,id:Log-forwarding-instructions,panelIndex:8,row:1,size_x:2,size_y:3,type:visualization),(col:1,columns:!(app_name,level,message),id:All-logs,panelIndex:9,row:4,size_x:12,size_y:7,sort:!('@timestamp',asc),type:search)),query:(query_string:(analyze_wildcard:!t,query:'{traceId}')),title:logs-dashboard,uiState:())
 
   docker-compose.yml 中给的 jhipster-registry 是本地模式的,可以根据注释部分内容,改成从git拉。好处是维护方便,坏处是,容易造成单点故障。使用git模式,就可以将 registry-config-sidekick 部分去掉。 
jhipster使用 liquibase 进行数据库版本管理,便于数据库版本变更记录管理和迁移。(rancher server 也用的liquibase)
把docker-compose.yml和rancher-compose.yml贴到rancher上,就能创建一个应用 stack了。
不过,好像漏了点啥,少了CICD,rancher和docker的compsoe.yml有了,但是,还没构建镜像呢,镜像还没push到registry呢,对吧
我司用gitlab管理源码,我在docker hub上发布了一个汉化的gitlab https://hub.docker.com/r/gitlab/gitlab-ce/tags/,如果要用官方镜像,参见 https://hub.docker.com/r/gitlab/gitlab-ce/tags/
version: '2'
services:
  gitlab:
    mem_limit: 5368709120 #限制内存最大 5G = 5*1024*1024*1024
    image: anjia0532/gitlab-ce-zh:10.3.3-ce.0
    volumes:
    - /data/gitlab/config:/etc/gitlab
    - /data/gitlab/data:/var/opt/gitlab
    - /data/gitlab/log:/var/log/gitlab
    ports:
    - 80:80/tcp
    - 443:443/tcp
 
  参见文档 GitLab Continuous Integration (GitLab CI)
为啥不用jenkins?
这个萝卜白菜各有所爱,我是出于压缩技术栈的考虑,
老牌 sonatype nexus oss 可以管理 Bower,Docker,Git LFS,Maven,npm,NuGet,PyPI,Ruby Gems,Yum Proxy,功能丰富
GitLab Container Registry administration gitlab registry跟gitlab集成,不需要额外安装服务
harbor rancher应用商店就有,安装方便,号称企业级registry,功能强大。
还是那句话,看需求,我司有部署maven和npm的需要,所以用了nexus oss,顺便管理docker registry。
我司是从16年八九月份开始拆分单体服务,彼时国内spring cloud,微服务等相关资料较少,国内流行dubbo(那会已经断更1年多了,虽然现在复更,但是对其前景不太看好)
从17年开始,圈内讨论spring cloud的渐渐多起来了,同时市面上也有了介绍spring cloud的书籍,比如周立的 Spring Cloud与Docker微服务架构实战 , 翟永超的 Spring Cloud微服务实战 等
但是用了spring cloud后,感觉spring cloud太复杂了(如果用了jhipster情况会好点),并没有实现微服务的初衷
在这种情况下,16年,国外buoyant公司提出Service Mesh概念,基于scala创建了linkerd项目。
service mesh 的设想就是,让开发人员专注于业务,不再分心于基础设施。
目前主流框架
其中istio和conduit都不太成熟,而linkerd和envoy都有商用案例,较为成熟。长远来看,更看好 istio和conduit
在此 分享一个dubbo的老用户的利好消息,据说 dubbo3 将兼容2,并且支持service mesh,并且支持反应式编程。参见 重大革新! Dubbo 3.0来了
扩展阅读资料
官方文档|ServiceMesh服务网格Istio面板组件&设计目标
演讲实录 | Service Mesh 时代的选边与站队(附PPT下载)
Service Mesh:下一代微服务
Service Mesh 在华为公有云的实践
明星分分合合的洪荒点击量,微博Mesh服务化改造如何支撑?(附PPT下载)
需要根据公司、团队实际情况理性选择框架,目前service mesh还处于垦荒阶段,而spring cloud或者dubbo还没到彻底过时的程度,建议持续关注,不建议立刻上马,如果已经落地了相关的微服务技术,不要盲目跟风,在可接受学习成本和开发成本情况下,可以考虑研究一下service mesh。
如果使用的是spring框架的话,建议抛开spring cloud,直接spring boot+service mesh,更清爽一些
谢谢支持
  支付宝
 支付宝 
  微信
 微信