OPA 進階-分散式利器 Bundle
Bundle
是OPA
管理policy
和data
的一種方式。
OPA
實現的輕量級策略引擎,一開始就是為了雲原生環境的service
提供解耦的策略服務,分散式是必然要考慮的問題。
在Bundle api
的設計中,其實就全面考慮並體現了在分散式應用中如何更好的解耦策略引擎的管理。
比如:
- 如何做集中配置管理
- 如何動態更新策略
- 如何監控策略引擎節點的狀態以及決策日誌收集
有了這些功能,再加上其高效的策略描述語言Rego
,OPA
才真正稱得上是雲原生時代的通用策略引擎。
本文將帶大家簡單梳理一遍Bundle
的組織方式、管理 api、及監控方式。
考慮到一次性過完不易消化,文末會提供一個直接可實操的docker-compose
版本的demo
,將全面覆蓋本文細節
建議大家看完本文,本機執行去體驗一下,會有更直觀的理解。
Bundle 檔案組織方式
下面我們先來看下Bundle
的檔案組織方式
在Bundle
下的data
,只能被識別data.json
和data.yaml
的檔案, 而其上邊的目錄會作為其資料字首
如下邊roles/data.json
(bundle/example
作為一個bundle
),會將data.json
的資料掛在data.roles
節點下
cd bundle/example
tree -a
.
├── .manifest
├── bindings
│ └── data.json
├── main.rego
├── rbac.rego
└── roles
└── data.json
其中.manifest
檔案是Bundle
的一個可選的後設資料(metadata
)配置檔案
cat .manifest
{
"revision" : "9f160bcd446bf50b1b17b570c322198a68d8e106",
"roots": ["roles", "bindings","rbac","system"]
}
它的作用是宣告Bundle
的版本revision
及其下的路徑字首(roots: path prefix
)
roots
不僅規定了Bundle
應該有的路徑字首;在用Bundle api
(後邊會提到)更新檔案時,也會按其規定的路徑字首來更新檔案
然後bundle
也支援tarball
格式載入到server
例如opa run -b
的方式指定Bundle
cd bundle/example
tar -czf bundle.tar.gz .
opa run -b bundle.tar.gz
Tips: 關於如何在互動式命令列裡傳遞
input
。 之前非 bundle 使用opa run quick-start repl.input:quick-start/input.json
到 bundle 格式時,就需要構建repl/input/data.json
檔案格式作為輸入
具體可以用時參考文件bundle-file-format
opa server api
在瞭解Bundle
支援的管理 api 前,我們先看下opa server api
主要 api 如下:
type | 用途 |
---|---|
Data api | 查詢文件(能被輸出的規則、虛擬文件等) |
Policy api | 查詢策略 |
Query api | 執行命令 |
Compile api | 執行部分查詢計算(partial evaluate query ) |
Health api | 健康檢查 |
Metric api | 指標統計(prometheus 格式) |
下面我們以文件查詢(Data
)api 為例嘗試下:
我們先用之前quick-start
的程式碼起一個opa sever
opa run --server quick-start
(注意:opa server api
的路徑字首為/v1/
, 對應的,查詢 api 路徑字首為/v1/data/
,)
# 構造input輸出請求
cat <<EOF > v1-data-input.json
{
"input": $(cat quick-start/input.json)
}
EOF
# 查詢 example_rbac
curl -s http://0.0.0.0:8181/v1/data/example_rbac?pretty=true -d @v1-data-input.json
{
"result": {
"allow": true,
"role_has_permission": [
"widget-reader"
],
"user_has_role": [
"widget-reader"
]
}
}
Tips:不指定路徑時,預設路徑為
data.system.main
,這時輸入不需要包裹在input
key 內。 也可以使用--set
和--set-file
可以覆蓋配置檔案中的配置opa run --server --set=default_decision=example_rbac/allow/ quick-start
curl -s http://0.0.0.0:8181/ -d @quick-start/input.json
而且 Data 查詢也支援組合引數如explain
,metrics
,provenance
等,詳細檢視文件,這裡就不展開了。
Bundle 管理 api
Bundle
為了在分散式系統中更好的展現 OPA 的威力,提供了四種 Api:
- Bundles
用於策略分發,可以定時輪訓更新
Bundle
包 - Decision Logs 定期上傳日誌包,支援按大小分片,開啟後會有日誌 id,決策日誌可追溯
- Status
定期上傳服務狀態,包含
metrics
等資訊 - Discovery
服務發現,可以用於集中管理
OPA
的Bundle
配置,各個節點下載定期同步配置後,按配置去更新Bundle
如下圖:
這裡舉個帶註釋Bundle
的四種介面配置例子
(先掃一遍留個印象,具體使用時檢視文件,後邊會提供可實操的程式碼)
# opa/config-bundle.yaml
services:
# 定義服務,支援多個
- name: example_bundle
url: http://demo-server:8888/
labels:
app: myapp
bundles:
# 定義bundle, 支援多個
authz:
# bundle所處的服務
service: example_bundle
# 這裡指從resource處更新bundle檔案包,即:
# http://demo-server:8888/bundle/rbac.tar.gz
resource: bundle/rbac.tar.gz
polling:
# 300~600s間更新一次
min_delay_seconds: 300
max_delay_seconds: 600
decision_logs:
service: example_bundle
# partition_name為區分上傳地址,會跟到 /logs 後, 即:
# http://demo-server:8888/logs/bundle
# 注意上傳的是gzip日誌檔案
partition_name: bundle
reporting:
min_delay_seconds: 30
max_delay_seconds: 60
status:
service: example_bundle
# 即 http://demo-server:8888/status/bundle
partition_name: bundle
# 預設查詢路徑
default_decision: rbac/allow
Bundle 整合方式
這裡我們簡單過下整合方式
opa server 方式
執行方式很簡單如下:
opa run -s -a 0.0.0.0:8181 -c opa/config-bundle.yaml
執行後,opa server 會根據配置自動拉取Bundle
包:rbac.tar.gz
下載成功後啟動策略服務。同時定期上傳決策日誌和狀態給服務端(即:demo-server:8888
)
go lib 方式
使用 lib github.com/open-policy-agent/opa/rego
整合
關鍵程式碼舉例如下:
// 構建查詢,PrepareForEval可重用
var err error
query, err := rego.New(
rego.LoadBundle("./rbac.tar.gz"),
rego.Query("x = data.rbac.allow"),
).PrepareForEval(context.Background())
// 執行查詢
results, err := query.Eval(context.Background(), rego.EvalInput(input))
if err != nil {
fmt.Fatalln("Opa eval error:", err)
return
} else if len(results) == 0 {
fmt.Fatalln("Opa eval error: no result")
return
}
fmt.Println("Opa result:", results[0].Expressions[0].Value)
具體組織方式官方推薦的有下邊集中式和分散式這兩種:
推薦感興趣的同學再去看下官方 go 整合的 demo: example-api-authz-go
Bundle 的監控
opa server 支援metrics
, 而且是prometheus
格式的
所以配合prometheus
可以直接進行對其資料指標的監控,如下圖:
再配合grafana
的dashbord
可以更好的展示metrics
資料,如下圖:
Bundle in action
上邊說這麼多,不實際試一下怎麼知道Bundle
究竟如何呢?
這裡提供一個docker-compose
版的 demo 給大家去本地驗證嘗試
裡邊提供了三種Bundle
版本:
- opa-bundle
- opa-discovery
- demo-sever (go lib 整合)
也提供了兩種版本的monitor
- slim version
- advance version
程式碼見:NewbMiao/opa-koans/bundle
裡邊有詳細的操作文件,有問題可以在 Repo 裡提 issue
這個 Repo 包含了這一系列的OPA
教程,歡迎感興趣的同學 star
關注!
同時我在知乎也建了一個OPA 技術圈,也歡迎大家參與討論。
好了,到此,OPA
的基本教程就結束了。後邊再抽空結合官方的例子寫些實戰教程吧。
最後附上一個 Repo 中驗證 Bundle 的過程,大家也可以從這裡開始嘗試哦
文章首發公眾號:newbmiao
推薦閱讀:OPA 系列
- 加微信實戰群請加微信(註明:實戰群):gocnio
相關文章
- 圖解|搞定分散式?程式設計師進階之路圖解分散式程式設計師
- 分散式鏈路追蹤的利器——Zipkin分散式
- 分散式:分散式事務(CAP、兩階段提交、三階段提交)分散式
- Python 函式進階-高階函式Python函式
- 分散式架構 -- 學習路線圖(入門到進階)分散式架構
- 分散式系統訊息中介軟體——RabbitMQ的使用進階篇分散式MQ
- 【進階篇】基於 Redis 實現分散式鎖的全過程Redis分散式
- 分散式系統理論進階7:Paxos變種和優化分散式優化
- 函式的進階函式
- 前端進階-樣式前端
- 338、分散式高階篇總結分散式
- java程式設計師進階:Redis分散式技術問題集錦Java程式設計師Redis分散式
- Orleans - 1 .NET生態構建分散式系統的利器分散式
- 分散式事務利器——RocketMQ事務訊息的啟示分散式MQ
- Python函式的進階Python函式
- SSM視訊教程:Java進階SSM分散式專案實戰視訊教程SSMJava分散式
- docker使用Open Policy Agent(OPA)進行訪問控制Docker
- Python 函式進階-遞迴函式Python函式遞迴
- 京東雲推出微服務:分散式服務框架JDSF進入公測階段微服務分散式框架
- day 10 函式的進階函式
- Python進階07 函式物件Python函式物件
- 函式進階應用3函式
- Python 函式進階-迭代器Python函式
- pytest進階之fixture函式函式
- 分散式技術“上位”進行時分散式
- 分散式基礎,啥是兩階段提交?分散式
- 分散式事務(二)之兩階段提交分散式
- 分散式事務(二)之三階段提交分散式
- iOS-生成Bundle包-引入bundle-使用bundleiOS
- 前端進階-ES6函式前端函式
- 前端進階-執行時函式前端函式
- 09-Python之路---函式進階Python函式
- JavaScript正規表示式進階指南JavaScript
- python進階(17)偏函式partialPython函式
- 圖解分散式架構的演進圖解分散式架構
- Locust 進行分散式負載測試分散式負載
- [分散式][分散式鎖]淺談分散式鎖分散式
- Kotlin進階(二)中綴、內聯、高階函式Kotlin函式