与Mongodb有关的数据持久
taskmanager2项目中的目录结构:
data:image/s3,"s3://crabby-images/d8c62/d8c626a4930014d2ffb6e06f8c41d11dd67c0e31" alt=""
src/taskmanager2/data2/taskRepository.go
其中的内容为与mongodb中的集合task紧密相关的CRUD操作
package data2
import (
"time"
"gopkg.in/mgo.v2"
"gopkg.in/mgo.v2/bson"
"taskmanager2/models2"
)
type TaskRepository struct {
C *mgo.Collection
}
func (r *TaskRepository) Create(task *models2.Task) error {
obj_id := bson.NewObjectId()
task.Id = obj_id
task.CreatedOn = time.Now()
task.Status = "Created"
err := r.C.Insert(&task)
return err
}
func (r *TaskRepository) Update(task *models2.Task) error {
// partial update on MogoDB
err := r.C.Update(bson.M{"_id": task.Id},
bson.M{"$set": bson.M{
"name": task.Name,
"description": task.Description,
"due": task.Due,
"status": task.Status,
"tags": task.Tags,
}})
return err
}
func (r *TaskRepository) Delete(id string) error {
err := r.C.Remove(bson.M{"_id": bson.ObjectIdHex(id)})
return err
}
func (r *TaskRepository) GetAll() []models2.Task {
var tasks []models2.Task
iter := r.C.Find(nil).Iter()
result := models2.Task{}
for iter.Next(&result) {
tasks = append(tasks, result)
}
return tasks
}
func (r *TaskRepository) GetById(id string) (task models2.Task, err error) {
err = r.C.FindId(bson.ObjectIdHex(id)).One(&task)
return
}
func (r *TaskRepository) GetByUser(user string) []models2.Task {
var tasks []models2.Task
iter := r.C.Find(bson.M{"createdby": user}).Iter()
result := models2.Task{}
for iter.Next(&result) {
tasks = append(tasks, result)
}
return tasks
}
task资源的处理函数
data:image/s3,"s3://crabby-images/be33a/be33a8be5f170b4e9954a07ba85a267b47000241" alt=""
资源路径与其对应的处理函数:
src/taskmanager2/routers2/task.go
package routers2
import (
"gopkg.in/gin-gonic/gin.v1"
"taskmanager2/controllers2"
"taskmanager2/common2"
)
// SetTaskRoutes configures routes for task entity
func SetTaskRoutes(router *gin.Engine) *gin.Engine {
taR := router.Group("/tm2/tasks")
taR.Use(common2.Authorize())
{
taR.POST("", controllers2.CreateTask)
taR.PUT(":id", controllers2.UpdateTask)
taR.DELETE(":id", controllers2.DeleteTask)
taR.GET("", controllers2.GetTasks)
taR.GET("t/:id/", controllers2.GetTaskByID)
taR.GET("users/:email/", controllers2.GetTasksByUser)
}
return router
}
具体函数:
data:image/s3,"s3://crabby-images/6404a/6404acdb1289706e6ba857fa359e3ca65b431d10" alt=""
src/taskmanager2/controllers2/taskCtrler.go
package controllers2
import (
"net/http"
"gopkg.in/mgo.v2"
"gopkg.in/mgo.v2/bson"
"gopkg.in/gin-gonic/gin.v1"
"taskmanager2/common2"
"taskmanager2/data2"
)
// CreateTask insert a new Task document
// Handler for HTTP Post - "/tasks
func CreateTask(c *gin.Context) {
var dataResource TaskResource
// Decode the incoming Task json
err := c.BindJSON(&dataResource)
if err != nil {
common2.DisplayAppError(
c,
err,
"Invalid Task data",
500,
)
return
}
task := &dataResource.Data
context := NewContext()
defer context.Close()
context.User = task.CreatedBy
col := context.DbCollection("tasks")
repo := &data2.TaskRepository{C: col}
// Insert a task document
repo.Create(task)
c.JSON(http.StatusCreated, TaskResource{Data: *task})
}
// GetTasks returns all Task document
// Handler for HTTP Get - "/tasks"
func GetTasks(c *gin.Context) {
context := NewContext()
defer context.Close()
col := context.DbCollection("tasks")
repo := &data2.TaskRepository{C: col}
tasks := repo.GetAll()
c.JSON(http.StatusOK, TasksResource{Data: tasks})
}
// GetTaskByID returns a single Task document by id
// Handler for HTTP Get - "/tasks/t/:id"
func GetTaskByID(c *gin.Context) {
// Get id from the incoming url
id := c.Param("id")
context := NewContext()
defer context.Close()
col := context.DbCollection("tasks")
repo := &data2.TaskRepository{C: col}
task, err := repo.GetById(id)
if err != nil {
if err == mgo.ErrNotFound {
c.JSON(http.StatusNoContent, gin.H{})
} else {
common2.DisplayAppError(
c,
err,
"An unexpected error has occurred",
500,
)
}
return
}
c.JSON(http.StatusOK, TaskResource{Data: task})
}
// GetTasksByUser returns all Tasks created by a User
// Handler for HTTP Get - "/tasks/users/:email"
func GetTasksByUser(c *gin.Context) {
// Get id from the incoming url
user := c.Param("email")
context := NewContext()
defer context.Close()
col := context.DbCollection("tasks")
repo := &data2.TaskRepository{C: col}
tasks := repo.GetByUser(user)
c.JSON(http.StatusOK, TasksResource{Data: tasks})
}
// UpdateTask update an existing Task document
// Handler for HTTP Put - "/tasks/:id"
func UpdateTask(c *gin.Context) {
// Get id from the incoming url
id := bson.ObjectIdHex(c.Param("id"))
var dataResource TaskResource
// Decode the incoming Task json
err := c.BindJSON(&dataResource)
if err != nil {
common2.DisplayAppError(
c,
err,
"Invalid Task data",
500,
)
return
}
task := &dataResource.Data
task.Id = id
context := NewContext()
defer context.Close()
col := context.DbCollection("tasks")
repo := &data2.TaskRepository{C: col}
// Update an existing Task document
if err := repo.Update(task); err != nil {
common2.DisplayAppError(
c,
err,
"An unexpected error has occurred",
500,
)
return
}
c.Status(http.StatusNoContent)
}
// DeleteTask deelete an existing Task document
// Handler for HTTP Delete - "/tasks/:id"
func DeleteTask(c *gin.Context) {
id := c.Param("id")
context := NewContext()
defer context.Close()
col := context.DbCollection("tasks")
repo := &data2.TaskRepository{C: col}
// Delete an existing Task document
err := repo.Delete(id)
if err != nil {
common2.DisplayAppError(
c,
err,
"An unexpected error has occurred",
500,
)
return
}
c.Status(http.StatusNoContent)
}
json资源模型
src/taskmanager2/controllers2/resources.go
package controllers2
import (
"taskmanager2/models2"
)
type (
...
TaskResource struct {
Data models2.Task `json:"data"`
}
TasksResource struct {
Data []models2.Task `json:"data"`
}
...
)
task资源的测试API操作
登录系统并获取jwt
post --- localhost:8890/tm2/users/login
调用:
data:image/s3,"s3://crabby-images/f37db/f37db15ff8e8dd0181e429f33601c008df853c8c" alt=""
响应:
data:image/s3,"s3://crabby-images/d023a/d023a1244a8ed6989ab0eb86d5fbebd7b832b1d9" alt=""
增加数据
post --- localhost:8890/tm2/tasks/
调用:
data:image/s3,"s3://crabby-images/3b8d1/3b8d1bc665e61fb83bbce2812a123cee9f9c1f43" alt=""
响应:
data:image/s3,"s3://crabby-images/5e630/5e630b106c0c6609382125be356e4859c7f24717" alt=""
数据库中查看结果:
data:image/s3,"s3://crabby-images/f1199/f119959c51795fbb62471132db8dafaf3268972d" alt=""
删除数据
delete --- localhost:8890/tm2/tasks/593e72625a9c4078ae5f1a79
调用:
data:image/s3,"s3://crabby-images/77549/77549913d761ecc1ba779a5b45ea66a7f9d226d4" alt=""
响应:
data:image/s3,"s3://crabby-images/a775b/a775bae8fcfc8c58824ac9ce8d3c053ccf6a88e3" alt=""
数据库中查看结果(第14条数据已被删除):
data:image/s3,"s3://crabby-images/5b7eb/5b7eb18c72a4006daf82a62423e19114bbf32ab5" alt=""
修改数据
put --- localhost:8890/tm2/tasks/593e72495a9c4078ae5f1a78
调用:
data:image/s3,"s3://crabby-images/e68bb/e68bba6fc6e872c5b9284bae564acacfd6fbf819" alt=""
响应:
data:image/s3,"s3://crabby-images/33ae8/33ae83a5d6743ea191c63689f0218684790061eb" alt=""
数据库中查看结果:
原数据:
data:image/s3,"s3://crabby-images/0d3c0/0d3c02e5a9918bc9d9c8754913908c69a073ad0e" alt=""
新数据:
data:image/s3,"s3://crabby-images/5d179/5d179447f60301626c3c28c4b0cafbe0f0888432" alt=""
获取全部的数据
get --- localhost:8890/tm2/tasks
调用:
data:image/s3,"s3://crabby-images/0728c/0728c9514aedfe0f5d3fb0d905a636eb48f91fd1" alt=""
响应:
data:image/s3,"s3://crabby-images/2d4f9/2d4f99e206ce6080f711a37958943f74b962d22a" alt=""
通过task的id获取数据
get --- localhost:8890/tm2/tasks/t/593e6f985a9c4078ae5f1a77/
调用:
data:image/s3,"s3://crabby-images/ce18f/ce18f69fe84744126c46549fd7cd5b6194bd0f7d" alt=""
响应:
data:image/s3,"s3://crabby-images/e3f75/e3f7536cdaa8ffd61541b566281cd4491b6c5529" alt=""
通过user的email获取数据
get --- localhost:8890/tm2/tasks/users/007@abc.com
调用:
data:image/s3,"s3://crabby-images/bf46e/bf46e83e8d9e1d636633f00e0ec66b4632c3d25f" alt=""
响应:
data:image/s3,"s3://crabby-images/a8c42/a8c42935d5fd8a71499f7ec5aaddecd563bfceba" alt=""
小结
至此,RESTful API中,user和task资源的功能已全部实现。实践证明gin框架能够很好地满足此应用的改造要求。
在改造的过程中,某些细节之处仍有不足,待日后慢慢完善。
网友评论