Openstack_通用技术_RPC 远程异步调用
目录
RPC
RPC: 同一个项目内的不同服务进程之间的交互方式。为不同的进程服务提供了 call()(同步) 和 cast()(异步) 两种调用方式。
问题 1: 在一个 Openstack 项目中拥有多个不同的进程服务,EG. API Service/Manage Service。 当我们通过 Client 发送 API Request 去调用 Manage Service 执行一个操作任务时,我们会希望这个调用的结果是能够快速响应到 Client 的(保证用户体验)。
问题 2: 而且进程服务之间的调用我们还需要考虑如何有效的避免进程服务之间调用的阻塞问题。EG. API Service 调用 Manage Service 时,如果不能及时的将 API Service 释放掉,那么 API Request 就会因为被占用,而无法处理新的请求。
对于上面两个问题,我们可以通过将具体的执行过程和响应过程分离来达到理想的效果。这也是 RPC 和 API 存在的原因之一。
一个通过 HTTP Request 调用操作函数的 RPC 实现样例
包含了下列两个过程实现:
- 接收 HTTP Request
- RPC 调用操作函数
环境
Devstack 用的 L 版,样例在自定义的 Openstack 项目中实现,详见自动化生成 Openstack 新项目开发框架。
接收 HTTP Request
- 定义 HTTP Request URL 路由转发的 Resource
详见Openstack Restful API 开发框架 Paste + PasteDeploy + Routes + WebOb 和
Openstack Paste.ini 文件详解
# project/api/v1/router.py
from project.api.v1 import new_resource
class APIRouter(octopunch.api.openstack.APIRouter):
"""Routes requests on the API to the appropriate controller and method."""
ExtensionManager = extensions.ExtensionManager
def _setup_routes(self, mapper, ext_mgr):
self.resources['versions'] = versions.create_resource()
mapper.connect("versions", "/",
controller=self.resources['versions'],
action='show')
mapper.redirect("", "/")
self.resources['new_resource'] = new_resource.create_resource(ext_mgr)
mapper.resource('new_resource', 'new_resource',
controller=self.resources['new_resource'])
- 将 Resource 和 HTTP 方法绑定到 Controller 的 Action 函数中
# project/api/v1/new_resource.py
class NewResourceController(wsgi.Controller):
def __init__(self, ext_mgr):
self.ext_mgr = ext_mgr
super(NewResourceController, self).__init__()
# Create() 对应了 HTTP 的 POST 方法
@wsgi.serializers()
def create(self, req, body):
"""Create a NewResource."""
context = req.environ['project.context']
# Sync the project database.
self.new_resource_api.db_sync(context)
def create_resource(ext_mgr):
"""project resource factory method."""
return wsgi.Resource(NewResourceController(ext_mgr))
上述两个文件实现了通过 HTTP Request 的 POST 方法来执行指定的 Create() 函数。
RPC 调用具体的操作函数
- 通过 API 调用 RPC-API
# project/new_resource/api.py
# import rpcapi module
from project.new_resource import rpcapi as new_resource_rpcapi
class API(base.Base):
"""API for interacting with the new_resource manager."""
def __init__(self, db_driver=None, image_service=None):
self.new_resource_rpcapi = new_resource_rpcapi.NewResourceAPI()
super(API, self).__init__(db_driver)
# 定义调用 rpcapi 的接口函数
def db_sync(self, context):
"""Call rpc api to start db sync for new_resource database."""
self.new_resource_rpcapi.db_sync(context)
API 的存在是为了能够快速的响应请求,至于之后的执行过程交由 RPC-API 和 Manager 来处理
- rpc-api.py 调用 manager.py
在 rpcapi.py 定义的 RPC 接口函数会自动的映射到 manager.py 中指定的处理函数。
# project/new_resource/rpcapi.py
class NewResourceAPI(object):
def __init__(self):
super(NewResourceAPI, self).__init__()
target = messaging.Target(topic=CONF.manage_topic,
version=self.RPC_API_VERSION)
serializer = objects_base.ProjectObjectSerializer()
self.client = rpc.get_client(target, version_cap='1.8',
serializer=serializer)
# 定义 rpcapi 函数,使用 cast 异步调用方式
def db_sync(self, context):
cctxt = self.client.prepare()
# 指定 rpcapi 的调用方式,和指定映射到 manager.py 的处理函数
cctxt.cast(context, 'db_sync')
RPC-API 的存在是为了快速的响应进程服务之间的调用请求。
# project/new_resource/manager.py
class NewResourceManager(manager.Manager):
RPC_API_VERSION = '1.8'
target = messaging.Target(version=RPC_API_VERSION)
def __init__(self, service_name=None, *args, **kwargs):
super(NewResourceManager, self).__init__(*args, **kwargs)
self._startup_delay = True
def init_host_with_rpc(self):
eventlet.sleep(CONF.periodic_interval)
self._startup_delay = False
def db_sync(self, context):
print "这里是具体的 RPC 操作函数"
小结:
Openstack 的 PRC 调用的过程为: api.py ⇒ rpcapi.py ⇒ manager.py
详见:Openstack Nova 源码分析 — RPC 远程调用过程
测试
- 启动 API 服务
project-api --config-file /etc/project/proname.conf
- 启动 Manager 服务
project-manager --config-file /etc/project/project.conf
- 发送 HTTP 请求
curl -i 'http://<Service_host_ip>:<service_port>/v1/<project_id>/<ResourceUrl>' -X POST -H "Content-Type: application/json" -H "X-Auth-Project-Id: admin" -H "X-Auth-Token: <token_id>" -d '<body_content_dict>'
注意: 样例需要根据自身开发环境进行调整,
Openstack_通用技术_RPC 远程异步调用的更多相关文章
- [SAP ABAP开发技术总结]Function远程、同步、异步调用
声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...
- XML-RPC远程方法调用
一.简介 XML-RPC的全称是XML Remote Procedure Call,即XML远程方法调用. 它是一套允许运行在不同操作系统.不同环境的程序实现基于Internet过程调用的规范和一系列 ...
- PRC远程过程调用
RPC(Remote Promote Call) 一种进程间通信方式.允许像调用本地服务一样调用远程服务. RPC框架的主要目标就是让远程服务调用更简单.透明.RPC框架负责屏蔽底层的传输方式(TCP ...
- Dubbo学习笔记4:服务消费端泛化调用与异步调用
本文借用dubbo.learn的Dubbo API方式来解释原理. 服务消费端泛化调用 前面我们讲解到,基于Spring和基于Dubbo API方式搭建简单的分布式系统时,服务消费端引入了一个SDK二 ...
- RabbitMQ九:远程过程调用RPC
定义 RPC(Remote Procedure Call Protocol)——远程过程调用协议:它是一种通过网络从远程计算机程序上请求服务,而不需要了解底层网络技术的协议.RPC协议假定某些传输协议 ...
- 抓到Dubbo异步调用的小BUG,再送你一个贡献开源代码的机会
hello,大家好呀,我是小楼. 最近一个技术群有同学at我,问我是否熟悉Dubbo,这我熟啊~ 他说遇到了一个Dubbo异步调用的问题,怀疑是个BUG,提到BUG我可就不困了,说不定可以水,哦不.. ...
- C# 多线程详解 Part.02(UI 线程和子线程的互动、ProgressBar 的异步调用)
我们先来看一段运行时会抛出 InvalidOperationException 异常的代码段: private void btnThreadA_Click(object sender, ...
- ABAP中的同步和异步调用
ABAP 的 CALL FUNCTION 类似于 Java/.NET 中的本地或远程方法调用.CALL FUNCTION 可以分为四种:1. Synchronous RFC (sRFC) - 同步调用 ...
- Ajax异步调用使用
//验证通知号重复 function checkinformcodeagage() { var informcode = $("#txtinformcode").val(); if ...
随机推荐
- 04-Django-templates
# 模板系统 - 模板:一组相同或者相似的页面,在需要个性化的地方进行留白,需要的时候只是用数据填充就可以使用 - 步骤: 1. 在settings中进行设置:TEMPLATES 2. 在tmeplt ...
- HTML面试问题收集(1)
1.浏览器页面有哪三层构成,分别是什么,作用是什么? 构成:结构层.表示层.行为层分别是:HTML.CSS.JavaScript 作用:HTML实现页面结构,CSS完成页面的表现与风格,JavaScr ...
- npm发布包
一.发布一个新包第一步:进入要发布的项目根目录,初始化为npm包: npm init 依次按提示填入包名.版本.描述.github地址.关键字.license等 这步完成之后会生成一个package. ...
- tomcat 启动日志乱码,idea中运行Tomcat也出现中文乱码:“淇℃伅”
打开到tomcat安装目录下的conf/文件夹 修改logging.properties文件, 找到 java.util.logging.ConsoleHandler.encoding = utf-8 ...
- Apache 用户认证
基本认证(Basic) 摘要认证(Digest) 更安全 创建一个名为 users 的认证口令: htpasswd -c /usr/local/apache2/conf/users sam ...
- CentOS 7 查看硬盘情况
用命令: lsblk 查看分区和磁盘 df -h 查看整 ...
- bean获取Spring容器
Person.java public class Person implements ApplicationContextAware{ private String name; private int ...
- 基于python3环境下搭建Robot Framework 自动化测试框架(一)
大家都知道,Robot Framework 是基于python2 环境 的一套自动化测试工具,据说python 2 到2020年不维护,现在用python 3 的环境搭建Robot Framework ...
- 【Luogu5293】[HNOI2019] 白兔之舞
题目链接 题目描述 略 Sol 考场上暴力 \(O(L)\) 50分真良心. 简单的推一下式子,对于一个 t 来说,答案就是: \[\sum_{i=0}^{L} [k|(i-t)] {L\choose ...
- :last-child----represents the last element among a group of sibling elements.
CSS伪类 :last-child 代表在一群兄弟元素中的最后一个元素. 举个例子: 从代码和图可以看出:last-child选择了最后一个li标签. 同时,我们换另外一段代码,看看是否还有这样的效果 ...