Vapor奇幻之旅(05 Fluent)

在上一篇Vapor奇幻之旅(04Routing)中我介紹了Routing的寫法拣宏,作為一個(gè)web應(yīng)用零聚,數(shù)據(jù)庫是必不可少的谆吴,而Fluent則是管理數(shù)據(jù)的一個(gè)抽象層站削,可以支持?jǐn)?shù)據(jù)庫的增刪改查等操作坊萝,默認(rèn)的FluentProvider支持sqlite數(shù)據(jù)庫,也就是說在沒有任何數(shù)據(jù)庫配置的情況下,可以通過Fluent Provider中的內(nèi)存數(shù)據(jù)庫來快速加載SQLite數(shù)據(jù)庫十偶,這樣做的好處是可以輕松的進(jìn)行接口測試菩鲜。

目前Vapor支持的數(shù)據(jù)庫如下:

數(shù)據(jù)庫類型 Key Package Class 是否來自官方
Memory memory Fluent Provider Fluent.MemoryDriver Yes
SQlite sqlite Fluent Provider Fluent.SQLiteDriver Yes
MySQL mysql MySQLProvider MySQLDriver.Driver Yes
PostgreSQL postgresql PostgreSQLProvider PostgreSQLDriver.Driver No
MongoDB N/A MongoProvider N/A No

對于大型數(shù)據(jù)庫官方只有支持到MySQL,稍顯遺憾,開發(fā)團(tuán)隊(duì)最近都在進(jìn)行Vapor 3的開發(fā)惦积,相信不久后就可以有更多的數(shù)據(jù)庫類型支持了接校,而且由于Fluent的抽象的特性,只要有相應(yīng)的驅(qū)動(dòng)狮崩,適配任何數(shù)據(jù)庫我想只是時(shí)間問題蛛勉。

既然是抽象層,我們先不管用啥數(shù)據(jù)庫睦柴,可以先把我們的數(shù)據(jù)模型搭建起來董习。

我想給我的網(wǎng)站加一段名人名言,于是我創(chuàng)建一個(gè)名為Quotes的模型爱只,代碼如下:

import Vapor
import FluentProvider
import HTTP

/// 名人名言
final class Quotes: Model {
    
    // 這個(gè)屬性能讓Fluent存儲(chǔ)額外的信息皿淋,如這個(gè)model的id
    let storage = Storage()
    
    //***下面是表中的屬性***
    
    /// 作者
    let author: String
    /// 內(nèi)容
    let content: String
    /// 描述
    let description: String
    
    /// 數(shù)據(jù)庫中列的名字
    struct Keys {
        static let id = "id"
        static let author = "author"
        static let content = "content"
        static let description = "description"
    }
    
    // MARK: 初始化Fluent
    
    /// 初始化Quotes
    required init(row: Row) throws {
        author = try row.get(Quotes.Keys.author)
        content = try row.get(Quotes.Keys.content)
        description = try row.get(Quotes.Keys.description)
    }
    
    // 序列化Quotes到數(shù)據(jù)庫
    func makeRow() throws -> Row {
        var row = Row()
        try row.set(Quotes.Keys.author, author)
        try row.set(Quotes.Keys.content, content)
        try row.set(Quotes.Keys.description, description)
        return row
    }

}

我們的model有了,下面就該聯(lián)系一下數(shù)據(jù)庫了恬试,F(xiàn)luent 提供了一個(gè)Preparation協(xié)議窝趣,源碼如下:

/// A preparation prepares the database for
/// any task that it may need to perform during runtime.
public protocol Preparation {

    /// The prepare method should call any methods
    /// it needs on the database to prepare.
    static func prepare(_ database: Database) throws

    /// The revert method should undo any actions
    /// caused by the prepare method.
    ///
    /// If this is impossible, the `PreparationError.revertImpossible`
    /// error should be thrown.
    static func revert(_ database: Database) throws
}

其中prepare方法是讓數(shù)據(jù)庫做好準(zhǔn)備的方法,比如新建table训柴,而revert方法則是對prepare做的操作進(jìn)行回滾操作哑舒,比如刪除table。

另外幻馁,JSON也是網(wǎng)絡(luò)通訊常用的數(shù)據(jù)格式洗鸵,模型通常也需要轉(zhuǎn)換為JSON串,或者需要解析json串到模型仗嗦。JSON庫為我們提供了JSONConvertible協(xié)議膘滨,demo如下

extension Quotes: JSONConvertible {
    convenience init(json: JSON) throws {
        self.init(
            author: try json.get(Quotes.Keys.author),
            content: try json.get(Quotes.Keys.content),
            description: try json.get(Quotes.Keys.description)
        )
    }
    
    func makeJSON() throws -> JSON {
        var json = JSON()
        try json.set(Quotes.Keys.id, id)
        try json.set(Quotes.Keys.author, author)
        try json.set(Quotes.Keys.content, content)
        try json.set(Quotes.Keys.description, description)
        return json
    }
}

在寫這個(gè)extension之前,還需要在mode里添加一個(gè)初始化方法:

/// 名人名言
final class Quotes: Model {
    ...
    // MARK: 初始化Fluent
    init(author: String, content: String, description: String) {
        self.author = author
        self.content = content
        self.description = description
    }
   ...
}

模型已經(jīng)建好了稀拐,那么作為一個(gè)數(shù)據(jù)庫模型火邓,怎么能少了增刪改查呢,藥藥藥德撬,切克鬧铲咨,增刪改查來一套:

這里我們需要開始寫Controller了,在controller文件夾內(nèi)創(chuàng)建一個(gè)QuotesController.swift的文件:

import Vapor
import FluentProvider

struct QuotesController {
    
    func addRoutes(to drop: Droplet) {
        let quots = drop.grouped("api","quots")
    }
}

然后在Config+Setup.swift中準(zhǔn)備好新創(chuàng)建的model:

private func setupPreparations() throws {
        preparations.append(Quotes.self)
}

接下來在創(chuàng)建一個(gè)Routers+Quotes.swift的文件并添加QuotesController的routs.
Routers+Quotes.swift:

import Vapor

extension Droplet {
    
    func setupQuotes() {
        let quotsController = QuotesController()
        quotsController.addRoutes(to: self)
    }
    
}

最后在Droplet+Setup.swift中添加setupQuotes()方法:

@_exported import Vapor

extension Droplet {
    public func setup() throws {
        setupQuotes()        
    }
}

現(xiàn)在就可以在我們的controller里面寫增刪改查了:

import Vapor
import FluentProvider

struct QuotesController {
    
    func addRoutes(to drop: Droplet) {
        let quots = drop.grouped("api","quots")
        //添加一個(gè)新的quots
        quots.post("create", handler: createQuots)
        //查詢所有的quotes
        quots.get(handler: allQuotes)
        // 更新quotes
        quots.post("update", handler: updateQuotes)
        // 刪除quotes
        quots.post("delete", handler: deleteQuotes)

    }

    /// 添加一個(gè)新的quots
    func createQuots(_ req: Request) throws -> ResponseRepresentable {
        guard let json = req.json else {
            throw Abort.badRequest
        }
        let quots = try Quotes(json: json)
        try quots.save()
        return quots
    }
    
    /// 查詢所有的quots
    func allQuotes(_ req: Request) throws -> ResponseRepresentable {
        let quots = try Quotes.all()
        return try quots.makeJSON()
    }
    /// 更新quotes
    func updateQuotes(_ req: Request) throws -> ResponseRepresentable {
        guard let json = req.json else {
            throw Abort.badRequest
        }
        
        let id: Int = try json.get("id")
        if let quots = try Quotes.find(id) {
            try quots.update(json: json)
        }
        
        return try Quotes.all().makeJSON()
    }
    
    // 刪除quotes
    func deleteQuotes(_ req: Request) throws -> ResponseRepresentable {
        guard let json = req.json else {
            throw Abort.badRequest
        }
        let id: Int = try json.get("id")
        if let quots = try Quotes.find(id) {
            try quots.delete()
        }
        
        return try Quotes.all().makeJSON()
    }
    
}

還需要在Quotes中加入一個(gè)update方法蜓洪,并把參數(shù)改成var

/// 名人名言
final class Quotes: Model {
    /// 作者
    var author: String
    /// 內(nèi)容
    var content: String
    /// 描述
    var description: String
    ...
}

extension Quotes {
    
    func update(json: JSON) throws {
        self.author = try json.get(Quotes.Keys.author)
        self.content = try json.get(Quotes.Keys.content)
        self.description = try json.get(Quotes.Keys.description)
        try self.save()
    }
    
}

現(xiàn)在我們的增刪改查就已經(jīng)完成了纤勒,下面cmd+r運(yùn)行程序,用Rested測試接口:

增加一個(gè)名言

查詢插入的結(jié)果
更新剛剛插入的數(shù)據(jù)
刪除剛剛插入的數(shù)據(jù)

由于默認(rèn)的數(shù)據(jù)庫是基于內(nèi)存加載的隆檀,重新運(yùn)行程序則會(huì)清空摇天,如果想要保存數(shù)據(jù)到服務(wù)器粹湃,你需要使用持續(xù)化的數(shù)據(jù)庫,如MySQL闸翅、PostgreSQL以及MongoDB再芋,后面我會(huì)對這幾個(gè)數(shù)據(jù)庫操作一一介紹。

關(guān)于Vapor其他知識(shí)坚冀,可以參考以下文章:

Vapor奇幻之旅(01開始)
Vapor奇幻之旅(02部署)
Vapor奇幻之旅(03上手)
Vapor奇幻之旅(04Routing)
Vapor奇幻之旅(05 Fluent)
Vapor奇幻之旅(06 PostgreSQL)
Vapor奇幻之旅(07 連接服務(wù)端PostgreSQL)
Vapor奇幻之旅(08 連接服務(wù)端MongoDB)
Vapor奇幻之旅(09 連接MySQL)

希望你對我的教程能夠喜歡济赎,你們的贊是我持續(xù)的動(dòng)力,歡迎加入QQ群參與互動(dòng):431296189

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末记某,一起剝皮案震驚了整個(gè)濱河市司训,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌液南,老刑警劉巖壳猜,帶你破解...
    沈念sama閱讀 222,865評論 6 518
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異滑凉,居然都是意外死亡统扳,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,296評論 3 399
  • 文/潘曉璐 我一進(jìn)店門畅姊,熙熙樓的掌柜王于貴愁眉苦臉地迎上來咒钟,“玉大人,你說我怎么就攤上這事若未≈熳欤” “怎么了?”我有些...
    開封第一講書人閱讀 169,631評論 0 364
  • 文/不壞的土叔 我叫張陵粗合,是天一觀的道長萍嬉。 經(jīng)常有香客問我,道長隙疚,這世上最難降的妖魔是什么壤追? 我笑而不...
    開封第一講書人閱讀 60,199評論 1 300
  • 正文 為了忘掉前任,我火速辦了婚禮甚淡,結(jié)果婚禮上大诸,老公的妹妹穿的比我還像新娘。我一直安慰自己贯卦,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,196評論 6 398
  • 文/花漫 我一把揭開白布焙贷。 她就那樣靜靜地躺著撵割,像睡著了一般。 火紅的嫁衣襯著肌膚如雪辙芍。 梳的紋絲不亂的頭發(fā)上啡彬,一...
    開封第一講書人閱讀 52,793評論 1 314
  • 那天羹与,我揣著相機(jī)與錄音,去河邊找鬼庶灿。 笑死纵搁,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的往踢。 我是一名探鬼主播腾誉,決...
    沈念sama閱讀 41,221評論 3 423
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼峻呕!你這毒婦竟也來了利职?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 40,174評論 0 277
  • 序言:老撾萬榮一對情侶失蹤瘦癌,失蹤者是張志新(化名)和其女友劉穎猪贪,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體讯私,經(jīng)...
    沈念sama閱讀 46,699評論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡热押,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,770評論 3 343
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了斤寇。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片桶癣。...
    茶點(diǎn)故事閱讀 40,918評論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖抡驼,靈堂內(nèi)的尸體忽然破棺而出鬼廓,到底是詐尸還是另有隱情,我是刑警寧澤致盟,帶...
    沈念sama閱讀 36,573評論 5 351
  • 正文 年R本政府宣布碎税,位于F島的核電站,受9級特大地震影響馏锡,放射性物質(zhì)發(fā)生泄漏雷蹂。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,255評論 3 336
  • 文/蒙蒙 一杯道、第九天 我趴在偏房一處隱蔽的房頂上張望匪煌。 院中可真熱鬧,春花似錦党巾、人聲如沸萎庭。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,749評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽驳规。三九已至,卻和暖如春署海,著一層夾襖步出監(jiān)牢的瞬間吗购,已是汗流浹背医男。 一陣腳步聲響...
    開封第一講書人閱讀 33,862評論 1 274
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留捻勉,地道東北人镀梭。 一個(gè)月前我還...
    沈念sama閱讀 49,364評論 3 379
  • 正文 我出身青樓,卻偏偏與公主長得像踱启,于是被迫代替她去往敵國和親报账。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,926評論 2 361

推薦閱讀更多精彩內(nèi)容

  • 發(fā)現(xiàn) 關(guān)注 消息 iOS 第三方庫禽捆、插件笙什、知名博客總結(jié) 作者大灰狼的小綿羊哥哥關(guān)注 2017.06.26 09:4...
    肇東周閱讀 12,125評論 4 61
  • #好好吃飯# #100天愛上營養(yǎng)早餐#76/100 減脂早餐 早餐前看到這本書里說到的"懸頁" [捂臉]原來"懸頁...
    Linda玲玲姐閱讀 537評論 0 0
  • 不該說從未,只能說好久沒覺得夜原來這么漫長了胚想。 今年我22歲琐凭,是人生的第二段短暫的愛情終結(jié)日。希望將來我會(huì)回過頭來...
    小粒仔閱讀 371評論 2 0