python 微服务_python微服务方案

本文介绍了一个基于Sanic框架的Python微服务架构方案。该方案利用Sanic的高性能特性,结合asyncio实现异步操作,使用uvloop提高并发能力,并集成多种流行库如asyncpg、aiohttp等,支持数据库操作、服务间通信及API标准化。
摘要由CSDN通过智能技术生成

介绍

使用python做web开发面临的一个最大的问题就是性能,在解决C10K问题上显的有点吃力。有些异步框架Tornado、Twisted、Gevent 等就是为了解决性能问题。这些框架在性能上有些提升,但是也出现了各种古怪的问题难以解决。

在python3.6中,官方的异步协程库asyncio正式成为标准。在保留便捷性的同时对性能有了很大的提升,已经出现许多的异步框架使用asyncio。

使用较早的异步框架是aiohttp,它提供了server端和client端,对asyncio做了很好的封装。但是开发方式和最流行的微框架flask不同,flask开发简单,轻量,高效。正是结合这些优点, 以Sanic为基础,集成多个流行的库来搭建微服务。 Sanic框架是和Flask相似的异步协程框架,简单轻量,并且性能很高。本项目就是以Sanic为基础搭建的python微服务框架。(思想适用于其他语言)

微服务设计原则个人总结:

86219ffb555cc2fb5c4b6608f1b2dc52.png

X 轴 :指的是水平复制,很好理解,就是讲单体系统多运行几个实例,做个集群加负载均衡的模式。

Z 轴 :是基于类似的数据分区,比如一个互联网打车应用突然或了,用户量激增,集群模式撑不住了,那就按照用户请求的地区进行数据分区,北京、上海、四川等多建几个集群。简单理解数据库拆分,比如分库分表

Y 轴 :就是我们所说的微服务的拆分模式,就是基于不同的业务拆分。

微服务总体架构:

ef808c0dcd64ef8c8d081f9f218b07d3.png

特点

  • 使用sanic异步框架,简单,轻量,高效。
  • 使用uvloop为核心引擎,使sanic在很多情况下单机并发甚至不亚于Golang。
  • 使用asyncpg为数据库驱动,进行数据库连接,执行sql语句执行。
  • 使用aiohttp为Client,对其他微服务进行访问。
  • 使用peewee为ORM,但是只是用来做模型设计和migration。
  • 使用opentracing为分布式追踪系统。
  • 使用unittest做单元测试,并且使用mock来避免访问其他微服务。
  • 使用swagger做API标准,能自动生成API文档。

服务端

使用sanic异步框架,有较高的性能,但是使用不当会造成blocking, 对于有IO请求的都要选用异步库。 添加库要慎重。sanic使用uvloop异步驱动,uvloop基于libuv使用Cython编写,性能比nodejs还要高。

功能说明:

启动前

  1. @app.listener('before_server_start')
  2. async def before_srver_start(app, loop):
  3. queue = asyncio.Queue()
  4. app.queue = queue
  5. loop.create_task(consume(queue, app.config.ZIPKIN_SERVER))
  6. reporter = AioReporter(queue=queue)
  7. tracer = BasicTracer(recorder=reporter)
  8. tracer.register_required_propagators()
  9. opentracing.tracer = tracer

app.db = await ConnectionPool(loop=loop).init(DB_CONFIG)

  • 创建DB连接池
  • 创建Client连接
  • 创建queue, 消耗span,用于日志追踪
  • 创建opentracing.tracer进行日志追踪

中间件

  1. @app.middleware('request')
  2. async def cros(request):
  3. if request.method == 'POST' or request.method == 'PUT':
  4. request['data'] = request.json
  5. span = before_request(request)
  6. request['span'] = span
  7. @app.middleware('response')
  8. async def cors_res(request, response):
  9. span = request['span'] if 'span' in request else None
  10. if response is None:
  11. return response
  12. result = {'code': 0}
  13. if not isinstance(response, HTTPResponse):
  14. if isinstance(response, tuple) and len(response) == 2:
  15. result.update({
  16. 'data': response[0],
  17. 'pagination': response[1]
  18. })
  19. else:
  20. result.update({'data': response})
  21. response = json(result)
  22. if span:
  23. span.set_tag('http.status_code', "200")
  24. if span:
  25. span.set_tag('component', request.app.name)
  26. span.finish()
  27. return response
  • 创建span, 用于日志追踪
  • 对response进行封装,统一格式

异常处理

对抛出的异常进行处理,返回统一格式

任务

创建task消费queue中对span,用于日志追踪

异步处理

由于使用的是异步框架,可以将一些IO请求并行处理

Example:

  1. async def async_request(datas):
  2. # async handler request
  3. results = await asyncio.gather(*[data[2] for data in datas])
  4. for index, obj in enumerate(results):
  5. data = datas[index]
  6. data[0][data[1]] = results[index]
  7. @user_bp.get('/')
  8. @doc.summary("get user info")
  9. @doc.description("get user info by id")
  10. @doc.produces(Users)
  11. async def get_users_list(request, id):
  12. async with request.app.db.acquire(request) as cur:
  13. record = await cur.fetch(
  14. """ SELECT * FROM users WHERE id = $1
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值