Search code examples
resthttpgogorillamux

gorilla/mux requests not matching URL pattern


I'm visiting some old code for a simple webserver I wrote some time ago, and my request patterns are no longer working. I have a function that initializes my routes as follows:

func (a *App) InitializeRoutes() {
    a.Router.HandleFunc("/api/forecast/detailed?city={city}&state={state}&period={period}", a.DetailedForecast).Methods("GET")
    a.Router.HandleFunc("/api/forecast/detailed?city={city}&state={state}", a.DetailedForecast).Methods("GET")
    a.Router.HandleFunc("/api/forecast/detailed?random", a.RandomDetailedForecast).Methods("GET")
    a.Router.HandleFunc("/api/forecast/hourly?city={city}&state={state}&hours={hours}", a.HourlyForecast).Methods("GET")
    a.Router.HandleFunc("/api/forecast/hourly?city={city}&state={state}", a.HourlyForecast).Methods("GET")
    a.Router.NotFoundHandler = http.HandlerFunc(a.Custom404Handler)
}

My custom 404 handler just returns a json of {"error": "Not Found"}

From here, I initialize these routes at the end of my app's initialize function (bottom 3 lines):

func (a *App) Initialize() {
    var err error
    conf.configure()
    sqlDataSource := fmt.Sprintf(
        "postgres://%s:%s@%s:%s/%s?sslmode=disable",
        conf.sqlUsername,
        conf.sqlPassword,
        conf.sqlHost,
        conf.sqlPort,
        conf.sqlDbName)
    a.DB, err = sql.Open("postgres", sqlDataSource)
    if err != nil {
        log.Fatal(err)
    }
    a.Redis = redis.NewClient(&redis.Options{
        Addr:     fmt.Sprintf("%s:%s", conf.redisHost, conf.redisPort),
        Password: conf.redisPassword,
        DB:       conf.redisDb,
    })
    a.Router = mux.NewRouter()
    a.Logger = handlers.CombinedLoggingHandler(os.Stdout, a.Router)
    a.InitializeRoutes()
}

My app runs on the following function:

func (a *App) Run() {
    port := fmt.Sprintf(":%s", os.Getenv("SERVER_PORT"))
    log.Fatal(http.ListenAndServe(port, a.Logger))
}

And my server runs on:

func main() {
    a := App{}
    a.Initialize()
    a.Run()
}

When I try and make a request to what as far as I can tell is a valid URL pattern, I get the following:

>>> import requests
>>> r = requests.get("http://localhost:8000/api/forecast/detailed?city=Chicago&state=IL")
>>> r
<Response [404]>
>>> r.json()
{'error': 'Not Found'}

EDIT

I made the change to move the query out of the URL. My routes now look like this:

a.Router.HandleFunc("/api/forecast/detailed", a.DetailedForecast).Queries("city", "{city:[a-zA-Z+]+}", "state", "{state:[a-zA-Z+]+}", "period", "{period:[a-zA-Z+]+}").Schemes("http").Methods("GET")
a.Router.HandleFunc("/api/forecast/detailed", a.DetailedForecast).Queries("city", "{city:[a-zA-Z+]+}", "state", "{state:[a-zA-Z+]+}").Schemes("http").Methods("GET")
a.Router.HandleFunc("/api/forecast/detailed/random", a.RandomDetailedForecast).Schemes("http").Methods("GET")
a.Router.HandleFunc("/api/forecast/hourly", a.HourlyForecast).Queries("city", "{city:[a-zA-Z+]+}", "state", "{state:[a-zA-Z+]+}", "hours", "{hours:[0-9]+}").Schemes("http").Methods("GET")
a.Router.HandleFunc("/api/forecast/hourly", a.HourlyForecast).Queries("city", "{city:[a-zA-Z+]+}", "state", "{state:[a-zA-Z+]+}").Schemes("http").Methods("GET")

When I make the same request, I am now seeing:

Get : unsupported protocol scheme ""

I've inspected the URL to see what is being sent:

2020/02/20 18:19:41 Scheme is:
2020/02/20 18:19:41 Host is:
2020/02/20 18:19:41 Path is: /api/forecast/detailed
2020/02/20 18:19:41 Query is: city=Chicago&state=IL

The path and the query look correct, but I do not understand why the scheme is invalid. My request is still r = requests.get("http://localhost:8000/api/forecast/detailed?city=Chicago&state=IL")


Solution

  • From the README... To match queries you need to use

    r := mux.NewRouter()
    ...
    r.Queries("key", "value")
    

    To match PathPrefix

    r.PathPrefix("/products/")
    

    I think you need to combine those two to achieve what you want.