Как получить выходные данные из request.RemoteAddr и X-AppEngine-Country, Region и т. Д. В Google App Engine Standard Env for Go?

У меня есть служба, работающая в стандартной среде Google App Engine, написанная на Go, которая настроена на использование последней среды выполнения при развертывании (api_version: go1 - в настоящее время Go 1.8).

В этом сервисе я проверяю заголовки запросов для различных целей.

func extractHeaders(res http.ResponseWriter, req *http.Request) {
    ctx := appengine.NewContext(req)
    clientIPAddress, _, _ := net.SplitHostPort(req.RemoteAddr) // Output is blank
    country := req.Header.Get("X-AppEngine-Country") // Output: US
    region := req.Header.Get("X-AppEngine-Region") // Output: ?
    city := req.Header.Get("X-AppEngine-City") // Output: ?
    cityLatLong := req.Header.Get("X-AppEngine-CityLatLong") // Output 0.000000,0.000000
    ...
}

Как видно из встроенного комментария для строк, начинающихся с того места, где я прочитал поле RemoteAddr, я не получаю результат, который я ожидал согласно документации AppEngine Standard, найденной здесь (Как обрабатываются запросы | Стандартная среда App Engine для Go | Google Cloud Platform).

Хотя в документации указано, что X-AppEngine- * не всегда можно заполнить на основе IP-адреса клиентского запроса, я никогда не вижу, чтобы они были заполнены данными, отличными от тех, которые я перечислил.

Требуется ли какая-либо конфигурация для заполнения этих заголовков (и поля RemoteAddr) в стандартной среде App Engine для Go? Я просто неправильно понимаю документацию?


person J. Wynn    schedule 19.01.2018    source источник


Ответы (1)


Следует отметить, что req.RemoteAddr получит IP-адрес последнего прокси в цепочке вашего приложения. В App Engine это, вероятно, будет 127.0.0.1. net.SplitHostPort, скорее всего, является ошибкой, поскольку IP-адрес может не иметь номера порта.

Данные региона, города и CityLatLong будут поступать с вашего IP-адреса, если эта информация доступна, возможно, ее нет, поскольку она не является строго обязательной. Что вы можете попробовать, так это вытащить IP-адрес из заголовка X-Forwarded-For. Обычно это дает вам полный список прокси и исходный IP-адрес (IPv4 или IPv6, см. Примеры результатов ниже).

Если бы вы использовали: package hello

import (
    "fmt"
    "net"
    "net/http"
)

func init() {
    http.HandleFunc("/", handler)
}

func handler(w http.ResponseWriter, r *http.Request) {
    clientIPAddress, _, err := net.SplitHostPort(r.RemoteAddr)
    if err != nil {
        clientIPAddress = err.Error()
    }
    rAddr := r.RemoteAddr
    forward := r.Header.Get("x-forwarded-for")
    country := r.Header.Get("X-AppEngineCountry")
    region := r.Header.Get("X-AppEngine-Region")
    city := r.Header.Get("X-AppEngine-City")
    cityLatLong := r.Header.Get("X-AppEngine-CityLatLong")
    fullHeader := r.Header
    fmt.Fprintf(w, "Hello, world!\n %v\n %v\n %v\n %v\n %v\n %v\n %v\n %v",
        clientIPAddress,
        rAddr,
        forward,
        country,
        region,
        city,
        cityLatLong,
        fullHeader
    )
}

Вы должны увидеть что-то похожее (пока доступны Region City и CityLatLong, в противном случае они будут пустыми):

Hello, world!
 missing port in address 127.0.0.1
 127.0.0.1
 2601:1c2:600:17b0::::, 2607:f8b0:400a:::2014
 ZZ
 or
 portland
 45.523452,-122.676207
 map[Accept-Encoding:[identity] X-Appengine-Default-Version-Hostname:
[127.0.0.1:8080] X-Appengine-User-Organization:[] X-Appengine-Citylatlong:
[45.523452,-122.676207] X-Appengine-Request-Log-Id:
[48f03baada92bbb3872a8b3a5baa3f8b0dfff413f64fbcfa] X-
Appengine-City:[portland] X-Forwarded-Proto:[https] X-Appengine-Server-Name:
[127.0.0.1] Via:[1.1 google] X-Appengine-Server-Software:[Development/2.0] 
X-Appengine-User-Id:[] X-Forwarded-For:
[2601:1c2:600:17b0:::: 2607:f8b0:400a:::2014] X-Goog-
Cloud-Shell-Target-Host:[127.0.0.1] X-Appengine-Country:[ZZ] Accept:
etc
etc

Надеюсь, это поможет.

person shindakun    schedule 20.01.2018
comment
Спасибо. Первоначально я пытался использовать заголовок x-forwarded-for, но он тоже возвращался пустым. Тем не менее, это прояснило мой вопрос относительно любой необходимой конфигурации, поскольку, похоже, ничего дополнительно не требуется. - person J. Wynn; 21.01.2018