Polaris(北极星)是名字服务协同团队合力研发的服务治理组件,具备服务注册、健康检查、服务发现、服务路由、负载均衡、故障节点熔断、动态权重调整与服务限流等功能。
Polaris Node.js SDK 采用微内核(Microkernel) + 插件(Plugins) 设计,可支持替换 11 种不同类型插件。
项目(包括依赖部分) 100% 由 TypeScript(JavaScript) 编写,不含任何 C/C++ 代码。
tnpm
$ tnpm install @dvsantd/polaris
yarn
$ yarn add @dvsantd/polaris --registry=https://mirrors.dvsantd.com/npm/
查询被调服务特定地址:
import { Consumer } from "@dvsantd/polaris";
const consumer = new Consumer();
const response = await (consumer.select("namespace", "service"));
if (response) {
const { instance: { host, port }} = response;
/**
* 通过 host, port 进行调用,
* 并调用 `update` 上报调用结果
*/
response.update(success, cost, code);
}
模块支持 Server
与 Agent
(暂未上线) 两种运行模式。
模式 | 优势 | 权衡 |
---|---|---|
Server(default) | 不需要部署 Agent 可替换不同类型插件 拥有更多配置 更好的性能 |
不能跨进程统计调用状态 |
Agent | 逻辑简单 支持跨进程统计调用状态 |
需要部署 Agent 可配置项较少 |
我应该如何选择运行模式?
在绝大多数情况下,直接使用默认的
Server
模式即可。由于模块只统计当前进程内的调用状态,如果服务调用量极少而又启动了 多进程 负载均衡时, 存在无法及时对实例进行调整(如屏蔽熔断)的可能,此时如果服务关心调用成功率,可以使用
Agent
模式。
具体切换方式可详见 名字服务插件 。
为了便于描述(表达)北极星组件的调用与返回结构,模块定义了如下几种数据类型:
实例 Instance
类型描述了服务节点详细信息,信息由静态与动态成员组成:
- 静态部分:
- id: 唯一 ID
- vpc_id: 腾讯云 VPC Id
- host: IP 或域名
- port: 端口号
- protocol: 协议信息
- staticWeight: 静态权重值, 取值 ∈ [0-1000]
- metadata: 元数据信息
- priority: 优先级
- version: 版本号
- logicSet: 逻辑区域
- location: 地理位置
- 动态部分:
- dynamicWeight: 动态权重值
- status: 当前状态
两个实例对象在进行比较时,直接比较 Instance.id
是否相同。也就是说,相同实例的 id
必须相同。
实例状态 InstanceStatus
描述了实例当前所处的状态:
- Normal: 正常
- HalfOpen: 半打开,各周期只选出极少次,负责探活
- HalfClose: 半关闭,不在任何模块中被选出,但计算调用结果
- Fused: 熔断,不在任何模块中被选出
各状态间迁移关系,可查看插件节 - 实例状态迁移
位置 Location
描述了一个特定的(地理)位置信息,其中包含:
- region
- zone
- campus(idc)
两个位置在比较时,按照范围由小至大进行匹配,campus
--> zone
--> region
一般用于就近调用等要求匹配位置的场景。
元数据 Metadata
以 {[key: string]: string} 形式存储特定对象(如实例对象)的描述信息。
一般用于规则路由等需进行对象筛选的场景。
import { Consumer } from "@dvsantd/polaris"
供服务调用方使用。
通过提供的主调(可选)与被调服务信息,按规则选出被调服务的一个特定实例或返回所有被调服务实例。
构造 Consumer
对象,用于获取被调方特定实例。
- service: 主调方服务信息或服务名(可选)
- plugins: 插件列表(可选)
- options: 配置参数(可选)
请注意:不要每次调用都构造一个 Consumer
对象,这样不仅会造成性能损耗,同时会导致无法按预期逻辑处理。如果你不得不这样做,请在使用完 Consumer
对象后调用 dispose()
接口将其释放。
获取被调服务中的一个特定实例,并返回上报对象。
const response = await consumer.select(namespace, service, metadata, args);
或
const response = await consumer.select(service, metadata, args);
- namespace: 被调服务名字空间(可选)
- service: 被调服务名
- metadata: 主调方服务元数据(可选)
- args: 本次调用附加参数(可选),在特定负载均衡(如一致性哈希)插件中使用。
response.instance
即为选出的被调服务的特定实例。
请保留 select(...)
接口返回的 response
对象,以便在调用完成后进行结果上报:
response.update(success, cost, code);
- success: 是否调用成功
- cost: 调用耗时(可选,默认为 0,单位为毫秒)
- code: 返回码(可选)
请注意:无论是否调用成功,在调用完成后一定要上报调用结果,否则模块将无法对实例进行动态调整。
获取被调方全部服务实例。
const instances = await consumer.list(namespace, service);
或
const instances = await consumer.list(service);
- namespace: 被调服务名字空间(可选)
- service: 被调服务名
instances
即为被调方全部服务实例。
强制刷新缓存。
const hasUpdated = await consumer.update(namespace, service, type);
- namespace: 被调服务名字空间
- service: 被调服务名
- type: 数据存储类别,为 RegistryCategory 枚举
如存在更新,则调用结果为 true
请留意:一般情况下,请同时更新 RegistryCategory.Instance
与 RegistryCategory.Rule
。
释放掉 Consumer
对象,在内部会释放掉相关的缓存和 socket 连接。
请注意:调用 dispose()
后,再调用 Consumer
对象的其他方法会抛出异常。
供服务提供方使用。
提供服务注册(注销)、心跳上报等能力。
构造 Provider
对象,用于获取被调方特定实例。
- plugins: 插件列表(可选)
请注意:不要每次调用都构造一个 Provider
对象,这样会造成性能损耗。
const response = await provider.register(namespace, service, token, instance, options);
- namespace: 命名空间
- service: 服务名
- token: 服务 Token 用来鉴权
- instance: 待注册的实例
- options: 注册选项(可选)
可通过 response
对注册的服务进行操作:
- id: 获取注册的实例
id
- unregister(): 服务注销
- heartbeat(): 心跳上报
const success = await provider.unregister(id, token);
- id: 实例 ID
- token: 服务 Token 用来鉴权
或
const success = await provider.unregister(namespace, service, host, port, token);
- namespace: 命名空间
- service: 服务名
- host: 节点 IP 或者域名
- port: 节点端口号
- token: 服务 Token 用来鉴权
可通过 success
判断注册是否成功。
const success = await provider.heartbeat(id, token);
- id: 实例唯一 ID
- token: 服务 Token 用来鉴权
或
const success = await provider.heartbeat(namespace, service, host, port, token)
- namespace: 命名空间
- service: 服务名
- host: 节点 IP 或者域名
- port: 节点端口号
- token: 服务 Token 用来鉴权
可通过 success
判断注册是否成功。
提供流量控制(整形)能力。
构造 Limiter
对象。
- plugins: 插件列表(可选)
- options: 配置参数(可选)
请注意:不要每次调用都构造一个 Limiter
对象,这样不仅会造成性能损耗,同时会导致无法按预期逻辑处理。
const response = await limiter.acquire(namespace, service, amount, cluster, labels, id)
- namespace: 命名空间
- service: 服务名
- amount: 申请配额的数量
- cluster: 集群名(可选)
- labels: 标签集合(可选)
- id: 上次调用返回 ID(对应
response.id
),用于二次获取配额时提升性能(可选)
response.quotas
即为申请的配额列表,而配额是否申请成功需等待其状态变更:
response.quotas[i].then((release) => { /** 配额申请成功 */ }, (err) => { /** 配额申请失败 */ });
当针对并发数进行限流时,在配额使用完成后,需调用 release()
方法释放对于配额的占用。
所有的内置插件被定义为实现了特定插件接口的 class
。缺省情况下,new Consumer()
时构造函数内部会使用默认参数实例化一系列插件实例。如果需要替换掉默认的插件或其实例化的参数,可以在 new Consumer()
时直接传入特定插件的实例。部分插件支持调用时参数,可以在调用 select()
方法时传入。
以负载均衡(一致性哈希)为例,其初始化和传参方法如下:
import { Consumer, HashRingLoadBalancer, plugins } from "@dvsantd/polaris";
const consumer = new Consumer({
[plugins.PluginType.LoadBalancer]: new HashRingLoadBalancer({
algorithm: 'md5'
})
});
(async () => {
await consumer.select("namespace", "service", metadata, {
[plugins.PluginType.LoadBalancer]: 'group/project'
})
})();
详情请看 插件
我们十分期待您的贡献,更多详情请参考 CONTRIBUTING.md