Способ решения этой проблемы - использование 2 разных URL, которые сначала обрабатываются traefik. Один URL-адрес используется для «direct grp c» (grp c .mydomain.com), другой - для grp c -web (назовем его grp c -web.mydomain.com). Traefik выполняет TLS-завершение для обоих.
Трафик grp c .mydomain.com c напрямую передается в контейнер, в котором запущен grpc_server. Трафик grp c -web.mydomain.com c передается посланнику, который действует как grp c -web-прокси, а затем передает трафик c серверу grpc_server.
Итак так как вы используете docker -compose, вам необходимо добавить службу envoy к вашему docker -compose.yml:
---
version: '3'
services:
traefik:
# traefik configuration from your question
# ...
grpc-server:
# grpc_server configuration from your question
# ...
envoy:
image: envoyproxy/envoy:v1.14.1
restart: unless-stopped
volumes:
- ./envoy.yaml:/etc/envoy/envoy.yaml
labels:
- traefik.enable=true
- traefik.http.routers.envoy.rule=Host(`grpc-web.mydomain.com`)
- traefik.http.services.envoy.loadbalancer.server.port=8080
- traefik.http.routers.envoy.tls=true
- traefik.http.routers.envoy.tls.certresolver=leresolver
Конфигурация envoy.yaml (смонтирована в разделе томов выше) выглядит так:
admin:
access_log_path: /tmp/admin_access.log
address:
socket_address: { address: 0.0.0.0, port_value: 9901 }
static_resources:
listeners:
- name: listener_0
address:
socket_address: { address: 0.0.0.0, port_value: 8080 }
filter_chains:
- filters:
- name: envoy.http_connection_manager
config:
codec_type: auto
stat_prefix: ingress_http
route_config:
name: local_route
virtual_hosts:
- name: local_service
domains: ["*"]
routes:
- match: { prefix: "/" }
route:
cluster: grpc_service
max_grpc_timeout: 0s
cors:
allow_origin_string_match:
- prefix: "*"
allow_methods: GET, PUT, DELETE, POST, OPTIONS
allow_headers: keep-alive,user-agent,cache-control,content-type,content-transfer-encoding,custom-header-1,x-accept-content-transfer-encoding,x-accept-response-streaming,x-user-agent,x-grpc-web,grpc-timeout
max_age: "1728000"
expose_headers: custom-header-1,grpc-status,grpc-message
http_filters:
- name: envoy.grpc_web
- name: envoy.cors
- name: envoy.router
clusters:
- name: grpc_service
connect_timeout: 0.25s
type: logical_dns
http2_protocol_options: {}
lb_policy: round_robin
hosts: [{ socket_address: { address: grpc-server, port_value: 8090 }}]
Это довольно простой c grp c веб-конфиг для посланника. Важно отметить, что мы установили address: grpc-server, port_value: 8090
в конфигурации кластерной конфигурации "grpc_service" на имя службы из docker -compose.yml и на порт, который прослушивает ваш grp c -сервер. Обратите внимание, что я переименовал ваш сервис из grpc_server в grp c -server, поскольку подчеркивание не является допустимым символом в именах хостов.
На стороне клиента используйте:
- "grp c -web.mydomain.com "в вашем javascript (grp c -web) коде.
- " grp c .mydomain.com "при написании клиента на другом языке (например, golang например).
Я создал рабочий пример, который можно найти в: https://github.com/rbicker/greeter