使用Golang和MongoDB构建 RESTful API

记录一下创建 RESTful API使用 Go开发语言和 MongoDB后台数据库

完整代码 Github

image

安装依赖

go get github.com/globalsign/mgo  // MongoDB的Go语言驱动
go get github.com/gorilla/mux   // Go语言的http路由库

API 结构预览

app.go

package main

import (
    "fmt"
    "net/http"

    "github.com/gorilla/mux"
)

func AllMovies(w http.ResponseWriter, r *http.Request) {
    fmt.Fprintln(w, "not implemented !")
}

func FindMovie(w http.ResponseWriter, r *http.Request) {
    fmt.Fprintln(w, "not implemented !")
}

func CreateMovie(w http.ResponseWriter, r *http.Request) {
    fmt.Fprintln(w, "not implemented !")
}

func UpdateMovie(w http.ResponseWriter, r *http.Request) {
    fmt.Fprintln(w, "not implemented !")
}

func DeleteMovie(w http.ResponseWriter, r *http.Request) {
    fmt.Fprintln(w, "not implemented !")
}

func main() {
    r := mux.NewRouter()

    r.HandleFunc("/movies", AllMovies).Methods("GET")
    r.HandleFunc("/movies/{id}", FindMovie).Methods("GET")
    r.HandleFunc("/movies", CreateMovie).Methods("POST")
    r.HandleFunc("/movies", UpdateMovie).Methods("PUT")
    r.HandleFunc("/movies", DeleteMovie).Methods("DELETE")

    http.ListenAndServe(":8080", r)
}

随着后续路由的增加,需要重构一个这个文件,仿照 beego的工程目录,我们也分别创建对应的 controllers
routes,改造一下上面的代码

  • 控制器

创建 controllers 文件夹和对应的文件 movies.go

movies.go

func AllMovies(w http.ResponseWriter, r *http.Request) {
    fmt.Fprintln(w, "not implemented !")
}

func FindMovie(w http.ResponseWriter, r *http.Request) {
    fmt.Fprintln(w, "not implemented !")
}

func CreateMovie(w http.ResponseWriter, t *http.Request) {
    fmt.Fprintln(w, "not implemented !")
}

func UpdateMovie(w http.ResponseWriter, r *http.Request) {
    fmt.Fprintln(w, "not implemented !")
}

func DeleteMovie(w http.ResponseWriter, r *http.Request) {
    fmt.Fprintln(w, "not implemented !")
}
  • 路由

创建一个 routes文件夹,并创建对应的文件 routes.go

routes.go

package routes

import (
    "net/http"

    "github.com/coderminer/restful/controllers"
    "github.com/gorilla/mux"
)

type Route struct {
    Method     string
    Pattern    string
    Handler    http.HandlerFunc
    Middleware mux.MiddlewareFunc
}

var routes []Route

func init() {
    register("GET", "/movies", controllers.AllMovies, nil)
    register("GET", "/movies/{id}", controllers.FindMovie, nil)
    register("POST", "/movies", controllers.CreateMovie, nil)
    register("PUT", "/movies", controllers.UpdateMovie, nil)
    register("DELETE", "/movies", controllers.DeleteMovie, nil)
}

func NewRouter() *mux.Router {
    r := mux.NewRouter()
    for _, route := range routes {
        r.Methods(route.Method).
            Path(route.Pattern).
            Handler(route.Handler)
        if route.Middleware != nil {
            r.Use(route.Middleware)
        }
    }
    return r
}

func register(method, pattern string, handler http.HandlerFunc, middleware mux.MiddlewareFunc) {
    routes = append(routes, Route{method, pattern, handler, middleware})
}

重构后的 app.go

package main

import (
    "net/http"

    "github.com/coderminer/restful/routes"
)

func main() {
    r := routes.NewRouter()

    http.ListenAndServe(":8080", r)
}

Models

  • 创建 models 文件夹和对应的文件 db.go(数据层),封装对MongoDB的封装
package models

import (
    "log"

    "github.com/globalsign/mgo"
)

const (
    host   = "127.0.0.1:27017"
    source = "admin"
    user   = "user"
    pass   = "123456"
)

var globalS *mgo.Session

func init() {
    dialInfo := &mgo.DialInfo{
        Addrs:    []string{host},
        Source:   source,
        Username: user,
        Password: pass,
    }
    s, err := mgo.DialWithInfo(dialInfo)
    if err != nil {
        log.Fatalln("create session error ", err)
    }
    globalS = s
}

func connect(db, collection string) (*mgo.Session, *mgo.Collection) {
    s := globalS.Copy()
    c := s.DB(db).C(collection)
    return s, c
}

func Insert(db, collection string, docs ...interface{}) error {
    ms, c := connect(db, collection)
    defer ms.Close()
    return c.Insert(docs...)
}

func FindOne(db, collection string, query, selector, result interface{}) error {
    ms, c := connect(db, collection)
    defer ms.Close()
    return c.Find(query).Select(selector).One(result)
}

func FindAll(db, collection string, query, selector, result interface{}) error {
    ms, c := connect(db, collection)
    defer ms.Close()
    return c.Find(query).Select(selector).All(result)
}

func Update(db, collection string, query, update interface{}) error {
    ms, c := connect(db, collection)
    defer ms.Close()
    return c.Update(query, update)
}

func Remove(db, collection string, query interface{}) error {
    ms, c := connect(db, collection)
    defer ms.Close()
    return c.Remove(query)
}

  • 业务逻辑层 models/movies.go
package models

import "github.com/globalsign/mgo/bson"

type Movies struct {
    Id          bson.ObjectId `bson:"_id" json:"id"`
    Name        string        `bson:"name" json:"name"`
    CoverImage  string        `bson:"cover_image" json:"cover_image"`
    Description string        `bson:"description" json:"description"`
}

const (
    db         = "Movies"
    collection = "MovieModel"
)

func (m *Movies) InsertMovie(movie Movies) error {
    return Insert(db, collection, movie)
}

func (m *Movies) FindAllMovies() ([]Movies, error) {
    var result []Movies
    err := FindAll(db, collection, nil, nil, &result)
    return result, err
}

func (m *Movies) FindMovieById(id string) (Movies, error) {
    var result Movies
    err := FindOne(db, collection, bson.M{"_id": bson.ObjectIdHex(id)}, nil, &result)
    return result, err
}

func (m *Movies) UpdateMovie(movie Movies) error {
    return Update(db, collection, bson.M{"_id": movie.Id}, movie)
}

func (m *Movies) RemoveMovie(id string) error {
    return Remove(db, collection, bson.M{"_id": bson.ObjectIdHex(id)})
}

控制器逻辑

  • CreateMovie
func CreateMovie(w http.ResponseWriter, r *http.Request) {
    defer r.Body.Close()
    var movie models.Movies
    if err := json.NewDecoder(r.Body).Decode(&movie); err != nil {
        responseWithJson(w, http.StatusBadRequest, "Invalid request payload")
        return
    }
    movie.Id = bson.NewObjectId()
    if err := dao.InsertMovie(movie); err != nil {
        responseWithJson(w, http.StatusInternalServerError, err.Error())
        return
    }
    responseWithJson(w, http.StatusCreated, movie)
}

使用 Postmancurl进行测试

  • AllMovies
func AllMovies(w http.ResponseWriter, r *http.Request) {
    defer r.Body.Close()
    var movies []models.Movies
    movies, err := dao.FindAllMovies()
    if err != nil {
        responseWithJson(w, http.StatusInternalServerError, err.Error())
        return
    }
    responseWithJson(w, http.StatusOK, movies)

}
  • FindMovie
func FindMovie(w http.ResponseWriter, r *http.Request) {
    vars := mux.Vars(r)
    id := vars["id"]
    result, err := dao.FindMovieById(id)
    if err != nil {
        responseWithJson(w, http.StatusInternalServerError, err.Error())
        return
    }
    responseWithJson(w, http.StatusOK, result)
}

http://127.0.0.1:8080/movies/5b45ef3a9d5e3e197c15e774

  • UpdateMovie
func UpdateMovie(w http.ResponseWriter, r *http.Request) {
    defer r.Body.Close()
    var params models.Movies
    if err := json.NewDecoder(r.Body).Decode(&params); err != nil {
        responseWithJson(w, http.StatusBadRequest, "Invalid request payload")
        return
    }
    if err := dao.UpdateMovie(params); err != nil {
        responseWithJson(w, http.StatusInternalServerError, err.Error())
        return
    }
    responseWithJson(w, http.StatusOK, map[string]string{"result": "success"})
}
  • DeleteMovie
func DeleteMovie(w http.ResponseWriter, r *http.Request) {
    vars := mux.Vars(r)
    id := vars["id"]
    if err := dao.RemoveMovie(id); err != nil {
        responseWithJson(w, http.StatusBadRequest, "Invalid request payload")
        return
    }

    responseWithJson(w, http.StatusOK, map[string]string{"result": "success"})
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,324评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,356评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,328评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,147评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,160评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,115评论 1 296
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,025评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,867评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,307评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,528评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,688评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,409评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,001评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,657评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,811评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,685评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,573评论 2 353

推荐阅读更多精彩内容