php小編草莓為您介紹CORS gRPC 網(wǎng)關(guān) GoLang。CORS是跨源資源共享的縮寫(xiě),是一種用于在瀏覽器中進(jìn)行跨域請(qǐng)求的機(jī)制。gRPC是一種高性能、開(kāi)源的遠(yuǎn)程過(guò)程調(diào)用(RPC)框架,可用于構(gòu)建分布式系統(tǒng)。而GoLang是一種強(qiáng)大的編程語(yǔ)言,具有高并發(fā)和簡(jiǎn)潔的特點(diǎn)。CORS gRPC 網(wǎng)關(guān) GoLang是將CORS和gRPC結(jié)合起來(lái),通過(guò)Go語(yǔ)言實(shí)現(xiàn)網(wǎng)關(guān),實(shí)現(xiàn)跨域請(qǐng)求的功能。在本文中,我們將詳細(xì)介紹CORS gRPC 網(wǎng)關(guān) GoLang的原理和使用方法,幫助您更好地理解和應(yīng)用該技術(shù)。
問(wèn)題內(nèi)容
我有一個(gè) vue.js 3 前端,我通過(guò) grpc-gateway
調(diào)用 golang 后端。我已經(jīng)這樣做了一段時(shí)間,但我看到了隧道盡頭的曙光。
我目前面臨 cors 問(wèn)題。然而,我正在閱讀有關(guān)如何處理它的相互矛盾的信息。因此,我想發(fā)帖,希望對(duì)大家有所幫助。
這里是我如何為 grpc(網(wǎng)關(guān))初始化 mux 服務(wù)器的代碼
func runhttpserver(server *http.server, httpendpoint, grpcendpoint, swaggerpath string) (err error) { server.addr = httpendpoint ctx, cancel := context.withcancel(context.background()) defer cancel() // register groc server endpoint mux := runtime.newservemux( runtime.witherrorhandler(func(ctx context.context, mux *runtime.servemux, marshaler runtime.marshaler, w http.responsewriter, r *http.request, err error, ) { s, ok := status.fromerror(err) if ok { if s.code() == codes.unavailable { err = status.error(codes.unavailable, errunavailable) } } runtime.defaulthttperrorhandler(ctx, mux, marshaler, w, r, err) }), ) opts := []grpc.dialoption{ grpc.withtransportcredentials(insecure.newcredentials()), grpc.withchainunaryinterceptor(), } if err = api.registerapiservicehandlerfromendpoint(ctx, mux, grpcendpoint, opts); err != nil { return } swmux := http.newservemux() swmux.handle("/", mux) serveswagger(swmux, swaggerpath) server.handler = swmux return server.listenandserve() }
登錄后復(fù)制
這里是我認(rèn)為應(yīng)該添加 cors 配置的地方,但我不確定這就是我在 server.go 文件中設(shè)置它的方式..
var httpserver http.server // run http server with grpc gateway g.go(func() error { fmt.println("starting http sever (port {}) and grpc gateway (port {})", strconv.itoa(cfg.server.httpport), strconv.itoa(cfg.server.grpcport), ) return rest.runhttpserver( &httpserver, ":"+strconv.itoa(cfg.server.httpport), ":"+strconv.itoa(cfg.server.grpcport), "/webapi", ) })
登錄后復(fù)制
控制臺(tái)錯(cuò)誤:
access to xmlhttprequest at 'http://localhost:8080/v1/test' from origin 'http://localhost:9000' has been blocked by cors policy: response to preflight request doesn't pass access control check: no 'access-control-allow-origin'
登錄后復(fù)制
我不知道在哪里添加類(lèi)似的內(nèi)容
func enablecors(w *http.responsewriter) { (*w).header().set("access-control-allow-origin", "*") }
登錄后復(fù)制
我覺(jué)得 golang grpc 網(wǎng)關(guān)應(yīng)該內(nèi)置一些東西,但我找不到任何東西?
如有任何建議,我們將不勝感激。
—–更新1—–
我已經(jīng)嘗試過(guò)
func enablecors(h http.handler) http.handler { return http.handlerfunc(func(w http.responsewriter, r *http.request) { w.header().set("access-control-allow-origin", "http://localhost:9000") w.header().set("access-control-allow-methods", "get, put, post, delete, head, options") h.servehttp(w, r) }) }
登錄后復(fù)制
和
func enablecors(h http.handler) http.handler { return http.handlerfunc(func(w http.responsewriter, r *http.request) { w.header().set("access-control-allow-origin", "*") w.header().set("access-control-allow-methods", "get, put, post, delete, head, options") h.servehttp(w, r) }) }
登錄后復(fù)制
和
func enablecors(h http.handler) http.handler { return http.handlerfunc(func(w http.responsewriter, r *http.request) { w.header().set("access-control-allow-origin", "http://localhost") w.header().set("access-control-allow-methods", "get, put, post, delete, head, options") h.servehttp(w, r) }) }
登錄后復(fù)制
結(jié)合
func serveSwagger(mux *http.ServeMux, swaggerPath string) { fileServer := http.FileServer(http.Dir(swaggerPath)) prefix := "/swagger-ui" mux.Handle(prefix, http.StripPrefix(prefix, fileServer)) }
登錄后復(fù)制
仍然有同樣的問(wèn)題..非常令人沮喪
解決方法
根據(jù)您在評(píng)論中提供的最新錯(cuò)誤:
從源“l(fā)ocalhost:9000”訪問(wèn)“l(fā)ocalhost:8080/v1/test”處的 xmlhttprequest 已被 cors 策略阻止:對(duì)預(yù)檢請(qǐng)求的響應(yīng)未通過(guò)訪問(wèn)控制檢查:它沒(méi)有 http 正常狀態(tài)。
您的瀏覽器正在發(fā)送預(yù)檢請(qǐng)求(options
http方法)以確定是否可以發(fā)出所需的跨源請(qǐng)求。
服務(wù)器正在響應(yīng)非 2xx 響應(yīng)。
我懷疑這是因?yàn)槟?enablecors
函數(shù)正在將請(qǐng)求傳播到 grpc-gateway 處理程序,該處理程序?qū)?options
http 方法不滿意并返回錯(cuò)誤狀態(tài),可能是:
< http/1.1 501 not implemented < content-type: application/json < vary: origin < date: fri, 25 nov 2022 11:17:52 gmt < content-length: 55 < {"code":12,"message":"method not allowed","details":[]}
登錄后復(fù)制
因此,為了避免這種情況,您希望在發(fā)出預(yù)檢請(qǐng)求時(shí)不進(jìn)一步傳播請(qǐng)求,例如
func enablecors(h http.handler) http.handler { return http.handlerfunc(func(w http.responsewriter, r *http.request) { w.header().set("access-control-allow-origin", "http://localhost:9000") w.header().set("access-control-allow-methods", "get, put, post, delete, head, options") if r.method == http.methodoptions { w.writeheader(http.statusnocontent) return } h.servehttp(w, r) }) }
登錄后復(fù)制
但是,上述內(nèi)容可能仍然不是 cors 處理的合理實(shí)現(xiàn)。您應(yīng)該為此使用現(xiàn)有的包,例如github.com/rs/cors,它將以合理的方式處理這個(gè)問(wèn)題,并處理任何潛在的陷阱等。
因此導(dǎo)入 github.com/rs/cors
然后執(zhí)行以下操作:
server.Handler = cors.AllowAll().Handler(swMux)
登錄后復(fù)制
應(yīng)該讓你開(kāi)始讓一切通過(guò)。該庫(kù)將允許您根據(jù)需要定制特定的來(lái)源、http 方法等。