go系列 通過Beego將之前實現的短url項目實現


通過Beego將之前實現的短url項目實現

 

正好通過這個小例子對之前了解的beego框架的基本內容進行一個簡單的應用

實現的完整代碼地址:https://github.com/pythonsite/go_simple_code/tree/master/beego_short_url

數據庫沒有什么變化,還是和之前一樣,主要是把處理邏輯放到beego中就可以了

代碼的主要目錄為:

復制代碼
localhost:beego_short_url zhaofan$ tree
.
├── beego_short_url
├── conf
│   └── app.conf
├── controllers
│   ├── default.go
│   └── short_url.go
├── main.go
├── models
│   └── data.go
├── routers
│   └── router.go
├── static
│   ├── css
│   ├── img
│   └── js
│       └── reload.min.js
├── tests
│   └── default_test.go
└── views
    └── index.tpl

10 directories, 10 files
復制代碼

關於長短url相互轉換的的請求和返回定義的struct在models下的data中,代碼為:

復制代碼
package models


type Long2ShortRequest struct {
    OriginUrl string `json:"origin_url"`
}

type ResponseHeader struct {
    Code int `json:"code"`
    Message string `json:"message"`
}

type Long2ShortResponse struct {
    ResponseHeader
    ShortUrl string `json:"short_url"`
}

type Short2LongRequest struct {
    ShortUrl string `json:"short_url"`
}

type Short2LongResponse struct {
    ResponseHeader
    OriginUrl string `json:"origin_url"`
}

type ShortUrl struct {
    ShortUrl string `json:"short_url" db:"short_url"`
}
復制代碼

而將原來在logic中的處理邏輯都放到了controllers中的short_url文件中

復制代碼
package controllers

import (
    "github.com/astaxie/beego"
    "beego_short_url/models"
    "encoding/json"
    "database/sql"
    "crypto/md5"
    "github.com/jmoiron/sqlx"
    "fmt"
    _ "github.com/go-sql-driver/mysql"
)

var (
    Db *sqlx.DB
)

func InitDb()(err error){
    Db, err = sqlx.Open("mysql",beego.AppConfig.String("Db::dsn"))
    if err != nil{
        beego.Error("connect to mysql failed:",err)
        return
    }
    return
}

type ShortUrl struct {
    Id int64 `db:"id"`
    ShortUrl string `db:"short_url"`
    OriginUrl string `db:"origin_url"`
    HashCode string `db:"hash_code"`
}

type ShortUrlController struct {
    beego.Controller
}


func (c *ShortUrlController) Jump() {
    shortUrl := c.GetString("shorturl")
    if len(shortUrl) == 0{
        return
    }
    var req models.Short2LongRequest
    var resp *models.Short2LongResponse = &models.Short2LongResponse{}

    defer func(){
        if err := recover();err != nil{
            beego.Error("panic err:",err)
            //resp.Code = 500
            //resp.Message = "server busy"
            //c.Data["json"] = resp
            //c.ServeJSON()
            return
        }
    }()
    req.ShortUrl = shortUrl
    resp,err := Short2Long(&req)
    if err != nil{
        beego.Error("short2Long failed error:",err)
        return
    }

    beego.Info("origin url:%s short url:%s",resp.OriginUrl,shortUrl)
    c.Redirect(resp.OriginUrl,301)
}

func (c *ShortUrlController) ShortUrlList() {
    limit,err := c.GetInt("limit")
    if err != nil{
        beego.Warn("not have limit params use default 10")
        limit = 10
    }
    data,err := GetLastShortUrl(limit)
    if err != nil{
        beego.Error("from db get url list error:",err)

    }

    for i,v:= range data{
        v.ShortUrl = fmt.Sprintf("/jump/?shorturl=%s",v.ShortUrl)
        data[i] = v
    }

    c.Data["url_list"] = data
    c.TplName = "index.tpl"
}

func(c *ShortUrlController) Long2Short(){
    var req models.Long2ShortRequest
    var resp *models.Long2ShortResponse = &models.Long2ShortResponse{}

    defer func(){
        if err := recover();err != nil{
            beego.Error("panic err:",err)
            resp.Code = 500
            resp.Message = "server busy"
            c.Data["json"] = resp
            c.ServeJSON()
            return
        }
    }()


    err := json.Unmarshal(c.Ctx.Input.RequestBody,&req)
    if err != nil{
        beego.Error("unmarshal failed,err:",err)
        resp.Code = 1001
        resp.Message = "json unmarshal failed"
        c.Data["json"] = resp
        c.ServeJSON()
        return
    }
    resp,err = Long2Short(&req)
    if err != nil{
        beego.Error("long2short failed,err:",err)
        resp.Code = 1002
        resp.Message = "long2short failed"
        c.Data["json"] = resp
        c.ServeJSON()
        return
    }
    c.Data["json"] = resp
    c.ServeJSON()



}

func(c *ShortUrlController) Short2Long(){
    var req models.Short2LongRequest
    var resp *models.Short2LongResponse = &models.Short2LongResponse{}

    defer func(){
        if err := recover();err != nil{
            beego.Error("panic err:",err)
            resp.Code = 500
            resp.Message = "server busy"
            c.Data["json"] = resp
            c.ServeJSON()
            return
        }
    }()


    err := json.Unmarshal(c.Ctx.Input.RequestBody,&req)
    if err != nil{
        beego.Error("unmarshal failed,err:",err)
        resp.Code = 1001
        resp.Message = "json unmarshal failed"
        c.Data["json"] = resp
        c.ServeJSON()
        return
    }
    resp,err = Short2Long(&req)
    if err != nil{
        beego.Error("Short2Long failed,err:",err)
        resp.Code = 1002
        resp.Message = "long2short failed"
        c.Data["json"] = resp
        c.ServeJSON()
        return
    }
    c.Data["json"] = resp
    c.ServeJSON()
}



func Long2Short(req *models.Long2ShortRequest) (response *models.Long2ShortResponse, err error) {
    response = &models.Long2ShortResponse{}
    urlMd5 := fmt.Sprintf("%x",md5.Sum([]byte(req.OriginUrl)))
    var short ShortUrl
    err = Db.Get(&short,"select id,short_url,origin_url,hash_code from short_url where hash_code=?",urlMd5)
    if err == sql.ErrNoRows{
        err = nil
        // 數據庫中沒有記錄,重新生成一個新的短url
        shortUrl,errRet := generateShortUrl(req,urlMd5)
        if errRet != nil{
            err = errRet
            return
        }
        response.ShortUrl = shortUrl
        return
    }
    if err != nil{
        return
    }
    response.ShortUrl = short.ShortUrl
    return
}

func generateShortUrl(req *models.Long2ShortRequest,hashcode string)(shortUrl string,err error){
    result,err := Db.Exec("insert INTO short_url(origin_url,hash_code)VALUES (?,?)",req.OriginUrl,hashcode)
    if err != nil{
        return
    }
    // 0-9a-zA-Z 六十二進制
    insertId,_:= result.LastInsertId()
    shortUrl = transTo62(insertId)
    _,err = Db.Exec("update short_url set short_url=? where id=?",shortUrl,insertId)
    if err != nil{
        fmt.Println(err)
        return
    }
    return
}

// 將十進制轉換為62進制   0-9a-zA-Z 六十二進制
func transTo62(id int64)string{
    // 1 -- > 1
    // 10-- > a
    // 61-- > Z
    charset := "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
    var shortUrl []byte
    for{
        var result byte
        number := id % 62
        result = charset[number]
        var tmp []byte
        tmp = append(tmp,result)
        shortUrl = append(tmp,shortUrl...)
        id = id / 62
        if id == 0{
            break
        }
    }
    fmt.Println(string(shortUrl))
    return string(shortUrl)
}


func Short2Long(req *models.Short2LongRequest) (response *models.Short2LongResponse, err error) {
    response = &models.Short2LongResponse{}
    var short ShortUrl
    err = Db.Get(&short,"select id,short_url,origin_url,hash_code from short_url where short_url=?",req.ShortUrl)
    if err == sql.ErrNoRows{
        response.Code = 404
        return
    }
    if err != nil{
        response.Code = 500
        return
    }
    response.OriginUrl = short.OriginUrl
    return
}


func GetLastShortUrl(limit int)(result []*models.ShortUrl,err error){
    err = Db.Select(&result,"select short_url from short_url ORDER BY id DESC  limit ? ",limit)
    return
}
復制代碼

在這里添加了一些之前沒有的功能:
獲取數據庫所有的short url 並且顯示在頁面上了,不過這里非常丑,如圖:

我們可以通過點擊相應的連接就會跳轉到長url的頁面
也可以通過模擬發送post請求來查看轉換的情況:

 

 

所有的努力都值得期許,每一份夢想都應該灌溉!


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM