極簡(jiǎn)istio wasmplugin go語(yǔ)言開發(fā)示例

前置條件

k8s集群,版本1.23.8

安裝方法略

istio安裝完成demo哭廉,版本1.17.2

curl -L https://git.io/getLatestIstio | ISTIO_VERSION=1.17.2 sh
cd istio-1.17.2/
./bin/istioctl install --set profile=demo -y
kubectl label namespace default istio-injection=enabled --overwrite

安裝httpbin,sleep 示例

httpbin主要為了提供http響應(yīng)妆棒,也可以用已有的能夠提供http服務(wù)的任意服務(wù)川抡。

sleep 為了在pod里使用curl命令,有別的pod中能執(zhí)行curl命令也可以边臼。

kubectl apply -f samples/httpbin/httpbin.yaml
kubectl apply -f samples/sleep/sleep.yaml

本地docker環(huán)境哄尔,版本 20.10.21

docker安裝略

go語(yǔ)言環(huán)境,版本1.18.1

go安裝略

阿里云或其他鏡像倉(cāng)庫(kù)

類似部署一個(gè)普通應(yīng)用柠并,wasm插件來源于某個(gè)鏡像倉(cāng)庫(kù)岭接。

為了方便使用了阿里云的免費(fèi)鏡像倉(cāng)庫(kù),也可以本地搭建或使用別的倉(cāng)庫(kù)臼予。

注意需要將阿里云的倉(cāng)庫(kù)設(shè)置為公開鸣戴!

示例功能

  • 在請(qǐng)求頭增加自定義的返回信息

  • 將插件的配置信息放到頭信息中返回(測(cè)試讀取配置的功能)

  • 簡(jiǎn)單的限流功能

直接看效果

使用我編譯好的鏡像直接部署,保存以下內(nèi)容為 gowasm.yaml

apiVersion: extensions.istio.io/v1alpha1
kind: WasmPlugin
metadata:
  name: my-go-wasm-plugin
  namespace: default
spec:
  selector:
    matchLabels:
      app: httpbin
  ## 編譯好的鏡像    
  url: oci://registry.cn-hangzhou.aliyuncs.com/hallywang/gowasm:20230530181612
  
  #插件的配置信息粘拾,在代碼中可以獲取到j(luò)son string
  pluginConfig:
    testConfig: abcddeeeee
    listconfig:
     - abc
     - def

部署到k8s中

kubectl apply -f gowasm.yaml

示例驗(yàn)證方法

  • 執(zhí)行以下命令窄锅,從sleep pod中發(fā)送http 請(qǐng)求到 httpbin ,打印出返回的header
SLEEP_POD=$(kubectl get pod -l app=sleep -o jsonpath={.items..metadata.name}

kubectl exec ${SLEEP_POD} -c sleep -- sh -c 'for i in $(seq 1 3); do curl --head -s httpbin:8000/headers; sleep 0.1; done'

  • 在未部署gowasm.yaml之前半哟,每次請(qǐng)求都會(huì)返回200成功酬滤,且頭信息中沒有自定義的內(nèi)容

  • 在部署了gowasm.yaml之后签餐,返回如下結(jié)果(兩次請(qǐng)求的結(jié)果)寓涨,有自定義的頭信息和403的返回說明插件部署成功。

    ## 第一次請(qǐng)求的返回結(jié)果:
    HTTP/1.1 200 OK
    server: envoy
    date: Wed, 31 May 2023 03:20:12 GMT
    content-type: application/json
    content-length: 526
    access-control-allow-origin: *
    access-control-allow-credentials: true
    x-envoy-upstream-service-time: 3
    ## 下面是插件增加的頭信息
    who-am-i: wasm-extension
    injected-by: istio-api!
    hally: wang
    wang: 1234567
    configdata: {"listconfig":["abc","def"],"testConfig":"abcddeeeee"}
    
    
    ## 第二次請(qǐng)求的返回結(jié)果:
    ## 限流起作用戒良,返回403
    HTTP/1.1 403 Forbidden
    powered-by: proxy-wasm-go-sdk!!
    content-length: 29
    content-type: text/plain
    who-am-i: wasm-extension
    injected-by: istio-api!
    hally: wang
    wang: 1234567
    configdata: {"listconfig":["abc","def"],"testConfig":"abcddeeeee"}
    date: Wed, 31 May 2023 03:20:12 GMT
    server: envoy
    x-envoy-upstream-service-time: 0
    
    

從代碼開始

安裝tinygo

tinygo可以將go編譯成wasm文件

https://tinygo.org/getting-started/install/

創(chuàng)建go工程

mkdir go-wasm-plugin
cd go-wasm-plugin
go mod init go-wasm

新增文件main.go

package main

import (
    "time"

    "github.com/tetratelabs/proxy-wasm-go-sdk/proxywasm"
    "github.com/tetratelabs/proxy-wasm-go-sdk/proxywasm/types"
)

func main() {
    proxywasm.SetVMContext(&vmContext{})
}

type vmContext struct {
    // Embed the default VM context here,
    // so that we don't need to reimplement all the methods.
    types.DefaultVMContext
}

// Override types.DefaultVMContext.
func (*vmContext) NewPluginContext(contextID uint32) types.PluginContext {
    return &pluginContext{}
}

type pluginContext struct {
    // Embed the default plugin context here,
    // so that we don't need to reimplement all the methods.
    types.DefaultPluginContext
    // the remaining token for rate limiting, refreshed periodically.
    remainToken int
    // // the preconfigured request per second for rate limiting.
    // requestPerSecond int
    // NOTE(jianfeih): any concerns about the threading and mutex usage for tinygo wasm?
    // the last time the token is refilled with `requestPerSecond`.
    lastRefillNanoSec int64
}

// Override types.DefaultPluginContext.
func (p *pluginContext) NewHttpContext(contextID uint32) types.HttpContext {
    return &httpHeaders{contextID: contextID, pluginContext: p}
}

type httpHeaders struct {
    // Embed the default http context here,
    // so that we don't need to reimplement all the methods.
    types.DefaultHttpContext
    contextID     uint32
    pluginContext *pluginContext
}

// Additional headers supposed to be injected to response headers.
var additionalHeaders = map[string]string{
    "who-am-i":    "go-wasm-extension",
    "injected-by": "istio-api!",
    "hally":       "wang",
    "wang":        "1234567",
    // 定義自定義的header糯崎,每個(gè)返回中都添加以上header
}

// 讀取部署yaml中的 pluginConfig 內(nèi)容几缭,用于插件的一些配置信息
var configData string

func (p *pluginContext) OnPluginStart(pluginConfigurationSize int) types.OnPluginStartStatus {
    proxywasm.LogDebug("loading plugin config")
    data, err := proxywasm.GetPluginConfiguration()
    if data == nil {
        return types.OnPluginStartStatusOK
    }

    if err != nil {
        proxywasm.LogCriticalf("error reading plugin configuration: %v", err)
        return types.OnPluginStartStatusFailed
    }

    // 插件啟動(dòng)的時(shí)候讀取配置
    configData = string(data)

    return types.OnPluginStartStatusOK
}

func (ctx *httpHeaders) OnHttpResponseHeaders(numHeaders int, endOfStream bool) types.Action {
    //添加headr
    for key, value := range additionalHeaders {
        proxywasm.AddHttpResponseHeader(key, value)
    }

    //為了便于演示觀察,將配置信息也加到返回頭里
    proxywasm.AddHttpResponseHeader("configData", configData)
    return types.ActionContinue
}

// 實(shí)現(xiàn)限流
func (ctx *httpHeaders) OnHttpRequestHeaders(int, bool) types.Action {
    current := time.Now().UnixNano()
    // We use nanoseconds() rather than time.Second() because the proxy-wasm has the known limitation.
    // TODO(incfly): change to time.Second() once https://github.com/proxy-wasm/proxy-wasm-cpp-host/issues/199
    // is resolved and released.
    if current > ctx.pluginContext.lastRefillNanoSec+1e9 {
        ctx.pluginContext.remainToken = 2
        ctx.pluginContext.lastRefillNanoSec = current
    }
    proxywasm.LogCriticalf("Current time %v, last refill time %v, the remain token %v",
        current, ctx.pluginContext.lastRefillNanoSec, ctx.pluginContext.remainToken)
    if ctx.pluginContext.remainToken == 0 {
        if err := proxywasm.SendHttpResponse(403, [][2]string{
            {"powered-by", "proxy-wasm-go-sdk!!"},
        }, []byte("rate limited, wait and retry."), -1); err != nil {
            proxywasm.LogErrorf("failed to send local response: %v", err)
            proxywasm.ResumeHttpRequest()
        }
        return types.ActionPause
    }
    ctx.pluginContext.remainToken -= 1
    return types.ActionContinue
}

新增文件 Dockerfile

# Dockerfile for building "compat" variant of Wasm Image Specification.
# https://github.com/solo-io/wasm/blob/master/spec/spec-compat.md

FROM scratch

COPY main.wasm ./plugin.wasm

編譯go代碼為wasm

go get

tinygo build -o main.wasm -scheduler=none -target=wasi main.go

編譯成功后,工程目錄中將出現(xiàn) main.wasm 文件

build一個(gè)Docker鏡像,推送到鏡像倉(cāng)庫(kù)

docker build . -t registy.cn-hangzhou.aliyuncs.com/USER_NAME/gowasm:0.1
docker push registy.cn-hangzhou.aliyuncs.com/USER_NAME/gowasm:0.1

新增部署yaml

gowasm.yaml

apiVersion: extensions.istio.io/v1alpha1
kind: WasmPlugin
metadata:
  name: my-go-wasm-plugin
  namespace: default
spec:
  selector:
    matchLabels:
      app: httpbin
  url: oci://registry.cn-hangzhou.aliyuncs.com/USER_NAME/gowasm:0.1
  pluginConfig:
    testConfig: abcddeeeee
    listconfig:
     - abc
     - def

部署到k8s,執(zhí)行測(cè)試腳本

kubectl apply -f gowasm.yaml

SLEEP_POD=$(kubectl get pod -l app=sleep -o jsonpath={.items..metadata.name}

kubectl exec ${SLEEP_POD} -c sleep -- sh -c 'for i in $(seq 1 3); do curl --head -s httpbin:8000/headers; sleep 0.1; done'

## 觀察返回內(nèi)容

刪除插件

kubectl delete wasmplugins my-go-wasm-plugin

遇到的問題

  • 修改代碼重新發(fā)布部署后薄霜,如果鏡像的tag沒變化某抓,可能出現(xiàn)不生效,這是因?yàn)閣asmplugin有自己的緩存機(jī)制,tag版本發(fā)生變化惰瓜,不會(huì)出現(xiàn)該問題

本文代碼

https://github.com/hallywang/go-wasm-plugin-example.git

其他玩法

  • istio官方提供的使用webassemblyhub來打包發(fā)布否副,內(nèi)容參考

https://istio.io/latest/zh/blog/2020/deploy-wasm-declarative/

但是發(fā)現(xiàn)webassemblyhub提供的工具不支持最新版本的istio备禀。

參考資料

https://tetrate.io/blog/istio-wasm-extensions-and-ecosystem/

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末奈揍,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子另患,更是在濱河造成了極大的恐慌,老刑警劉巖柴淘,帶你破解...
    沈念sama閱讀 212,542評(píng)論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件秘通,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡第股,警方通過查閱死者的電腦和手機(jī)话原,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,596評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門诲锹,熙熙樓的掌柜王于貴愁眉苦臉地迎上來涉馅,“玉大人,你說我怎么就攤上這事稚矿。” “怎么了桥爽?”我有些...
    開封第一講書人閱讀 158,021評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵昧识,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我缀去,道長(zhǎng)习霹,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,682評(píng)論 1 284
  • 正文 為了忘掉前任阎曹,我火速辦了婚禮煞檩,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘斟湃。我一直安慰自己,他們只是感情好凝赛,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,792評(píng)論 6 386
  • 文/花漫 我一把揭開白布墓猎。 她就那樣靜靜地躺著,像睡著了一般毙沾。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上寇仓,一...
    開封第一講書人閱讀 49,985評(píng)論 1 291
  • 那天,我揣著相機(jī)與錄音俭嘁,去河邊找鬼乳愉。 笑死屯远,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的慨丐。 我是一名探鬼主播,決...
    沈念sama閱讀 39,107評(píng)論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼备闲,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼捅暴!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起蓬痒,我...
    開封第一講書人閱讀 37,845評(píng)論 0 268
  • 序言:老撾萬榮一對(duì)情侶失蹤梧奢,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后亲轨,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,299評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡器虾,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,612評(píng)論 2 327
  • 正文 我和宋清朗相戀三年兆沙,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了晕粪。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,747評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖昏鹃,靈堂內(nèi)的尸體忽然破棺而出诀诊,到底是詐尸還是另有隱情,我是刑警寧澤属瓣,帶...
    沈念sama閱讀 34,441評(píng)論 4 333
  • 正文 年R本政府宣布抡蛙,位于F島的核電站,受9級(jí)特大地震影響粗截,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜熊昌,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,072評(píng)論 3 317
  • 文/蒙蒙 一婿屹、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧昂利,春花似錦、人聲如沸梯捕。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,828評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)碌奉。三九已至,卻和暖如春嫉拐,著一層夾襖步出監(jiān)牢的瞬間魁兼,已是汗流浹背婉徘。 一陣腳步聲響...
    開封第一講書人閱讀 32,069評(píng)論 1 267
  • 我被黑心中介騙來泰國(guó)打工盖呼, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人几晤。 一個(gè)月前我還...
    沈念sama閱讀 46,545評(píng)論 2 362
  • 正文 我出身青樓蟹瘾,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親憾朴。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,658評(píng)論 2 350

推薦閱讀更多精彩內(nèi)容