Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Golang MongoDB Error: result argument must be a slice address

Tags:

rest

mongodb

go

mgo

I have a function that needs to retrieve all users from a mongoDB collection which is throwing an error when I try to call it through a REST endpoint. The function is:

func (usercontroller UserController) GetAllUsers(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
        session := usercontroller.session.Copy()
        defer session.Close()

        // Stub user
        result := models.User{}

        // get all users
        if err := session.DB("Auth").C("users").Find(nil).All(&result); err != nil {
                w.WriteHeader(404)
                return
        }

        // Marshal provided interface into JSON structure
        mResult, _ := json.Marshal(result)

        // Write content-type, statuscode, and the data package
        w.Header().Set("Content-Type", "application/json")
        w.WriteHeader(200)
        fmt.Fprintf(w, "%s", mResult)
}

The error is:

2016/06/25 16:00:21 http: panic serving 127.0.0.1:49898: result argument must be a slice address
goroutine 13 [running]:
net/http.(*conn).serve.func1(0xc82005e380)
    /usr/lib/go-1.6/src/net/http/server.go:1389 +0xc1
panic(0x729b60, 0xc82000ba50)
    /usr/lib/go-1.6/src/runtime/panic.go:443 +0x4e9
gopkg.in/mgo%2ev2.(*Iter).All(0xc8201280f0, 0x704600, 0xc82007e360, 0x0, 0x0)
    /root/work/src/gopkg.in/mgo.v2/session.go:3696 +0x742
gopkg.in/mgo%2ev2.(*Query).All(0xc8200bc600, 0x704600, 0xc82007e360, 0x0, 0x0)
    /root/work/src/gopkg.in/mgo.v2/session.go:3723 +0x4f
Auth-REST-Server/controllers.UserController.GetAllUsers(0xc82000d040, 0x7f3c1b9bf950, 0xc820061c70, 0xc8200c6380, 0x0, 0x0, 0x0)
    /root/work/src/Auth-REST-Server/controllers/user.go:34 +0x251
Auth-REST-Server/controllers.(UserController).GetAllUsers-fm(0x7f3c1b9bf950, 0xc820061c70, 0xc8200c6380, 0x0, 0x0, 0x0)
    /root/work/src/Auth-REST-Server/server.go:21 +0x5f
github.com/julienschmidt/httprouter.(*Router).ServeHTTP(0xc820016380, 0x7f3c1b9bf950, 0xc820061c70, 0xc8200c6380)
    /root/work/src/github.com/julienschmidt/httprouter/router.go:344 +0x195
net/http.serverHandler.ServeHTTP(0xc820122000, 0x7f3c1b9bf950, 0xc820061c70, 0xc8200c6380)
    /usr/lib/go-1.6/src/net/http/server.go:2081 +0x19e
net/http.(*conn).serve(0xc82005e380)
    /usr/lib/go-1.6/src/net/http/server.go:1472 +0xf2e
created by net/http.(*Server).Serve
    /usr/lib/go-1.6/src/net/http/server.go:2137 +0x44e

What am I doing wrong here. I thought I could just call Find(nil).All(&result) and retrieve all of the records accordingly.

like image 238
ozfive Avatar asked Mar 11 '23 21:03

ozfive


1 Answers

In your code you have:

// Stub user
result := models.User{}

So result represents a single user, but you want all the users from the database, so the following should do the job:

// Stub user
result := make([]models.User, 0, 10) // Here you can specify a len and a cap.

This way, result is a slice of models.User.

Also, in Golang, you should never ignore an error:

// Marshal provided interface into JSON structure
mResult, _ := json.Marshal(result)

Should be:

// Marshal provided interface into JSON structure
mResult, err := json.Marshal(result)
if err != nil {
    // do something with the error (log it or write it in the response)
}

EDIT: I updated the code, sorry for that.

like image 177
jnmoal Avatar answered Mar 14 '23 15:03

jnmoal