KeystoneJS 源碼分析(1)

KeystoneJS 是一個優(yōu)秀的并且非常有個性的 Node CMS,數(shù)據(jù)庫使用 MongoDB官辈,而手頭的一個項目,因為主要數(shù)據(jù)庫便是 MongoDB拴念,結合當時的需求钧萍,評估后采用了 KeystoneJS 作為運營后臺的基礎。在較短的時間內(nèi)政鼠,達到了不錯的應用效果风瘦,滿足了初期的各項業(yè)務迭代需求。

應用過程中公般,感受到 KeystoneJS 的文檔算是優(yōu)秀万搔,但作為一個快速發(fā)展的開源項目,文檔中有些缺失也在所難免官帘,并且文檔偏重于純使用瞬雹,若有更多改造的需求,分析源碼便是個必修課刽虹。

為了更好的滿足后續(xù)的業(yè)務改進需求酗捌,便計劃分析下 KeystoneJS 的源碼,也希望從中學習優(yōu)秀框架的設計思想涌哲。


一. 沿調用棧(縱向)分析

1. 調用示例

  • keystone.js

    keystone.import('models')
    ...
    keystone.start()
    
  • models/Project.js

    var Project = new keystone.List('Project', {
        schema: { collection: 't_project' },
        label: '項目',
    })
    
    Project.add({
      name: { type: String, required: true, label: '項目名稱'},
      pv: { type: Number, label: 'PV' }
    })
    
    ...
    Project.register()
    

2. keystone 源碼中的核心類型

List

  • lib/list.js

    module.exports = function (keystone) {
      function List (key, options) {
        var defaultOptions = { 
          schema: {
                    collection: keystone.prefixModel(key),
                },
                ...
        };
        ...
        this.options = utils.options(defaultOptions, options)
        ...
        this.schema = new keystone.mongoose.Schema({}, this.options.schema);
        ...
      }
      
      ...
      // 筆記:依賴到 mongo Schema
          Object.defineProperty(this, 'nameIsVirtual', { 
            get: function () {
                return this.model.schema.virtuals[this.mappings.name] ? true : false
            } 
          })
      ...
      
      List.prototype.add = require('./list/add')
      ...
      List.prototype.field = require('./list/field')
      ...
      return List
    }
    
  • lib/list/add.js

    function add () {
      var add = function(obj, prefix) {
        ...
        var keys = Object.keys(obj)
        for(...keys) {
          // 筆記:addField(path, fieldOptions)
          addField(prefix + key, obj[key])
        }
      }
        
      var addField = function (path, options) {
        ...
        this.uiElements.push({
          type: 'field',
          field: this.field(path, options)
        })
      }.bind(this)
      
      var args = Array.prototype.slice.call(arguments)
      ...
      
      _forEach(args, function (def) {
        
        ...
        add(def);
      }
    }
    

Field (belongs to List)

  • lib/list/field.js

    function field (path, options) {
      ...
      if (options.type === String) {
        options.type = Field.Types.Number
      }
      ...
      
      // 筆記:new Field(list, path, options)
      var field = new options.type(this, path, options)
      ...
      return field
    }
    
    module.exports = field
    

Field Type

  • fields/types/number/NumberType.js

    var FieldType = require('../Type');
    ...
    
    function number (list, path, options) {
      this._nativeType = Number
      ...
    }
    
    util.inherits(number, FieldType)
    
  • fields/types/Type.js

    function Field (list, path, options) {
      ...
      this.addToSchema(this.list.schema)
      ...
    }
    
    ...
    
    Field.prototype.addToSchema = function (schema) {
      var ops = (this._nativeType) ? _.defaults({ type: this._nativeType }, this.options) : this.options;
      // 筆記:listSchema.path(field.path, field.options)
      schema.path(this.path, ops)
      ...
    }
    

二. 沿搜索(橫向)分析

1. 關鍵詞 schema.

  • 依賴 schema 的屬性/方法
    • add()
    • path()
    • nested[]
    • virtual()
    • pre()
    • methods
    • paths[]
    • statics
    • collection
    • url
    • mimetype
    • size
  • 依賴 schema 的模塊
    • List
      • list.js
      • list/add.js
      • list/buildSearchTextIndex.js
      • list/declaresTextIndex.js
      • list/expandColumns.js
      • list/register.js
    • 一些非基本 Field 類型的 field type, 如 NameType, PasswordType, SelectType, RelationType
    • 一些高級 Field 類型, 如 LocalFilesType, LocationType, MarkdownType
    • schemaPlugins
      • autokey
      • history
      • sortable
      • track
    • lib/storage.js

2. 關鍵詞 query.{ $(query 相關)

  • fields/types/number/NumberType.js
  • lib/core/createItem.js
  • lib/list/addSearchToQuery.js
  • lib/list/apiForGet.js
  • lib/list/pageinate.js
  • lib/middleware/api.js
  • lib/schemaPlugins/methods/getRelated.js
  • lib/view.js
  • fidleds/types/*Type.js
  • lib/list/addFiltersToQuery.js
  • lib/list/getSearchFilter.js

.exec

  • lib/core/createItems.js
  • lib/list/apiForGet.js
  • lib/list/getUniqueValue.js
  • lib/list/paginate.js
  • lib/list/updateItem.js
  • lib/schemaPlugins/autokey.js
  • lib/schemaPlugins/methods/getRelated.js
  • lib/schemaPlugins/sortable.js
  • lib/session.js
  • lib/view.js
最后編輯于
?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末胖缤,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子阀圾,更是在濱河造成了極大的恐慌哪廓,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,348評論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件初烘,死亡現(xiàn)場離奇詭異涡真,居然都是意外死亡,警方通過查閱死者的電腦和手機肾筐,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,122評論 2 385
  • 文/潘曉璐 我一進店門哆料,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人吗铐,你說我怎么就攤上這事剧劝。” “怎么了抓歼?”我有些...
    開封第一講書人閱讀 156,936評論 0 347
  • 文/不壞的土叔 我叫張陵讥此,是天一觀的道長拢锹。 經(jīng)常有香客問我,道長萄喳,這世上最難降的妖魔是什么卒稳? 我笑而不...
    開封第一講書人閱讀 56,427評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮他巨,結果婚禮上充坑,老公的妹妹穿的比我還像新娘。我一直安慰自己染突,他們只是感情好捻爷,可當我...
    茶點故事閱讀 65,467評論 6 385
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著份企,像睡著了一般也榄。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上司志,一...
    開封第一講書人閱讀 49,785評論 1 290
  • 那天甜紫,我揣著相機與錄音,去河邊找鬼骂远。 笑死囚霸,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的激才。 我是一名探鬼主播拓型,決...
    沈念sama閱讀 38,931評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼瘸恼!你這毒婦竟也來了劣挫?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 37,696評論 0 266
  • 序言:老撾萬榮一對情侶失蹤钞脂,失蹤者是張志新(化名)和其女友劉穎揣云,沒想到半個月后捕儒,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體冰啃,經(jīng)...
    沈念sama閱讀 44,141評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,483評論 2 327
  • 正文 我和宋清朗相戀三年刘莹,在試婚紗的時候發(fā)現(xiàn)自己被綠了阎毅。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,625評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡点弯,死狀恐怖扇调,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情抢肛,我是刑警寧澤狼钮,帶...
    沈念sama閱讀 34,291評論 4 329
  • 正文 年R本政府宣布碳柱,位于F島的核電站,受9級特大地震影響熬芜,放射性物質發(fā)生泄漏莲镣。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,892評論 3 312
  • 文/蒙蒙 一涎拉、第九天 我趴在偏房一處隱蔽的房頂上張望瑞侮。 院中可真熱鬧,春花似錦鼓拧、人聲如沸半火。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,741評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽钮糖。三九已至,卻和暖如春种玛,著一層夾襖步出監(jiān)牢的瞬間藐鹤,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,977評論 1 265
  • 我被黑心中介騙來泰國打工赂韵, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留娱节,地道東北人。 一個月前我還...
    沈念sama閱讀 46,324評論 2 360
  • 正文 我出身青樓祭示,卻偏偏與公主長得像肄满,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子质涛,可洞房花燭夜當晚...
    茶點故事閱讀 43,492評論 2 348

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

  • Spring Cloud為開發(fā)人員提供了快速構建分布式系統(tǒng)中一些常見模式的工具(例如配置管理稠歉,服務發(fā)現(xiàn),斷路器汇陆,智...
    卡卡羅2017閱讀 134,629評論 18 139
  • Spark SQL, DataFrames and Datasets Guide Overview SQL Dat...
    草里有只羊閱讀 18,303評論 0 85
  • 這個時間點很失落怒炸,失落,落毡代,落阅羹,落,快落到海底了教寂。 就沉到海底吧捏鱼,讓海水的壓力四年八方壓過來,壓到爆炸酪耕,壓到?jīng)]有感...
    呼吸到他存在閱讀 166評論 0 0
  • 偉大的牛群同志曾經(jīng)研究過三個定理,其中之一便是 一切物體在沒有受到力的作用時看尼,總保持勻速直線運動狀態(tài)或靜止狀態(tài)递鹉。 ...
    綿花不白閱讀 142評論 0 0