2014-01-26 17:46:25 -05:00
|
|
|
storage = require 'lib/storage'
|
|
|
|
|
2014-01-03 13:32:13 -05:00
|
|
|
class CocoSchema extends Backbone.Model
|
|
|
|
constructor: (path, args...) ->
|
|
|
|
super(args...)
|
|
|
|
@urlRoot = path + '/schema'
|
|
|
|
|
|
|
|
window.CocoSchema = CocoSchema
|
|
|
|
|
|
|
|
class CocoModel extends Backbone.Model
|
|
|
|
idAttribute: "_id"
|
|
|
|
loaded: false
|
|
|
|
loading: false
|
2014-01-26 17:46:25 -05:00
|
|
|
saveBackups: false
|
2014-01-03 13:32:13 -05:00
|
|
|
@schema: null
|
|
|
|
|
|
|
|
initialize: ->
|
|
|
|
super()
|
|
|
|
if not @constructor.className
|
|
|
|
console.error("#{@} needs a className set.")
|
|
|
|
@markToRevert()
|
|
|
|
if @constructor.schema?.loaded
|
|
|
|
@addSchemaDefaults()
|
|
|
|
else
|
|
|
|
@loadSchema()
|
2014-01-26 17:46:25 -05:00
|
|
|
@once 'sync', @onLoaded, @
|
|
|
|
@saveBackup = _.debounce(@saveBackup, 500)
|
2014-02-12 15:41:41 -05:00
|
|
|
|
2014-01-03 13:32:13 -05:00
|
|
|
type: ->
|
|
|
|
@constructor.className
|
|
|
|
|
2014-01-26 17:46:25 -05:00
|
|
|
onLoaded: ->
|
2014-01-03 13:32:13 -05:00
|
|
|
@loaded = true
|
|
|
|
@loading = false
|
|
|
|
@markToRevert()
|
2014-01-26 17:46:25 -05:00
|
|
|
if @saveBackups
|
|
|
|
existing = storage.load @id
|
|
|
|
if existing
|
2014-02-12 15:41:41 -05:00
|
|
|
@set(existing, {silent:true})
|
2014-01-26 17:46:25 -05:00
|
|
|
CocoModel.backedUp[@id] = @
|
2014-02-12 15:41:41 -05:00
|
|
|
|
2014-01-26 17:46:25 -05:00
|
|
|
set: ->
|
|
|
|
res = super(arguments...)
|
2014-02-05 14:03:32 -05:00
|
|
|
@saveBackup() if @saveBackups and @loaded and @hasLocalChanges()
|
2014-01-26 17:46:25 -05:00
|
|
|
res
|
2014-02-12 15:41:41 -05:00
|
|
|
|
2014-01-26 17:46:25 -05:00
|
|
|
saveBackup: ->
|
|
|
|
storage.save(@id, @attributes)
|
|
|
|
CocoModel.backedUp[@id] = @
|
2014-02-12 15:41:41 -05:00
|
|
|
|
2014-01-26 17:46:25 -05:00
|
|
|
@backedUp = {}
|
2014-01-03 13:32:13 -05:00
|
|
|
|
|
|
|
loadSchema: ->
|
|
|
|
unless @constructor.schema
|
|
|
|
@constructor.schema = new CocoSchema(@urlRoot)
|
|
|
|
@constructor.schema.fetch()
|
|
|
|
|
2014-02-12 15:41:41 -05:00
|
|
|
@constructor.schema.once 'sync', =>
|
2014-01-03 13:32:13 -05:00
|
|
|
@constructor.schema.loaded = true
|
|
|
|
@addSchemaDefaults()
|
|
|
|
@trigger 'schema-loaded'
|
|
|
|
|
|
|
|
@hasSchema: -> return @schema?.loaded
|
|
|
|
schema: -> return @constructor.schema
|
|
|
|
|
|
|
|
validate: ->
|
|
|
|
result = tv4.validateMultiple(@attributes, @constructor.schema?.attributes or {})
|
|
|
|
if result.errors?.length
|
|
|
|
console.log @, "got validate result with errors:", result
|
|
|
|
return result.errors unless result.valid
|
|
|
|
|
|
|
|
save: (attrs, options) ->
|
|
|
|
options ?= {}
|
|
|
|
success = options.success
|
|
|
|
options.success = (resp) =>
|
|
|
|
@trigger "save:success", @
|
|
|
|
success(@, resp) if success
|
|
|
|
@markToRevert()
|
2014-01-26 17:46:25 -05:00
|
|
|
@clearBackup()
|
2014-01-03 13:32:13 -05:00
|
|
|
@trigger "save", @
|
|
|
|
return super attrs, options
|
|
|
|
|
|
|
|
fetch: ->
|
2014-02-14 18:35:54 -05:00
|
|
|
super(arguments...)
|
2014-01-03 13:32:13 -05:00
|
|
|
@loading = true
|
|
|
|
|
|
|
|
markToRevert: ->
|
|
|
|
@_revertAttributes = _.clone @attributes
|
|
|
|
|
|
|
|
revert: ->
|
|
|
|
@set(@_revertAttributes, {silent: true}) if @_revertAttributes
|
2014-01-26 17:46:25 -05:00
|
|
|
@clearBackup()
|
2014-02-12 15:41:41 -05:00
|
|
|
|
2014-01-26 17:46:25 -05:00
|
|
|
clearBackup: ->
|
|
|
|
storage.remove @id
|
2014-01-03 13:32:13 -05:00
|
|
|
|
|
|
|
hasLocalChanges: ->
|
|
|
|
not _.isEqual @attributes, @_revertAttributes
|
|
|
|
|
|
|
|
cloneNewMinorVersion: ->
|
|
|
|
newData = $.extend(null, {}, @attributes)
|
|
|
|
new @constructor(newData)
|
|
|
|
|
|
|
|
cloneNewMajorVersion: ->
|
|
|
|
clone = @cloneNewMinorVersion()
|
|
|
|
clone.unset('version')
|
|
|
|
clone
|
|
|
|
|
|
|
|
isPublished: ->
|
|
|
|
for permission in @get('permissions') or []
|
|
|
|
return true if permission.target is 'public' and permission.access is 'read'
|
|
|
|
false
|
|
|
|
|
|
|
|
publish: ->
|
|
|
|
if @isPublished() then throw new Error("Can't publish what's already-published. Can't kill what's already dead.")
|
|
|
|
@set "permissions", (@get("permissions") or []).concat({access: 'read', target: 'public'})
|
|
|
|
|
|
|
|
addSchemaDefaults: ->
|
|
|
|
return if @addedSchemaDefaults or not @constructor.hasSchema()
|
|
|
|
@addedSchemaDefaults = true
|
|
|
|
for prop, defaultValue of @constructor.schema.attributes.default or {}
|
|
|
|
continue if @get(prop)?
|
|
|
|
#console.log "setting", prop, "to", defaultValue, "from attributes.default"
|
|
|
|
@set prop, defaultValue
|
|
|
|
for prop, sch of @constructor.schema.attributes.properties or {}
|
|
|
|
continue if @get(prop)?
|
|
|
|
#console.log "setting", prop, "to", sch.default, "from sch.default" if sch.default?
|
|
|
|
@set prop, sch.default if sch.default?
|
|
|
|
|
2014-02-15 20:29:54 -05:00
|
|
|
getReferencedModels: (data, schema, path='/', shouldLoadProjection=null) ->
|
2014-01-03 13:32:13 -05:00
|
|
|
# returns unfetched model shells for every referenced doc in this model
|
|
|
|
# OPTIMIZE so that when loading models, it doesn't cause the site to stutter
|
|
|
|
data ?= @attributes
|
|
|
|
schema ?= @schema().attributes
|
|
|
|
models = []
|
|
|
|
|
|
|
|
if $.isArray(data) and schema.items?
|
|
|
|
for subData, i in data
|
2014-02-15 20:29:54 -05:00
|
|
|
models = models.concat(@getReferencedModels(subData, schema.items, path+i+'/', shouldLoadProjection))
|
2014-01-03 13:32:13 -05:00
|
|
|
|
|
|
|
if $.isPlainObject(data) and schema.properties?
|
|
|
|
for key, subData of data
|
|
|
|
continue unless schema.properties[key]
|
2014-02-15 20:29:54 -05:00
|
|
|
models = models.concat(@getReferencedModels(subData, schema.properties[key], path+key+'/', shouldLoadProjection))
|
2014-01-03 13:32:13 -05:00
|
|
|
|
2014-02-15 20:29:54 -05:00
|
|
|
model = CocoModel.getReferencedModel data, schema, shouldLoadProjection
|
2014-01-03 13:32:13 -05:00
|
|
|
models.push model if model
|
|
|
|
return models
|
|
|
|
|
2014-02-15 20:29:54 -05:00
|
|
|
@getReferencedModel: (data, schema, shouldLoadProjection=null) ->
|
2014-01-03 13:32:13 -05:00
|
|
|
return null unless schema.links?
|
|
|
|
linkObject = _.find schema.links, rel: "db"
|
|
|
|
return null unless linkObject
|
|
|
|
return null if linkObject.href.match("thang_type") and not @isObjectID(data) # Skip loading hardcoded Thang Types for now (TODO)
|
|
|
|
|
|
|
|
# not fully extensible, but we can worry about that later
|
|
|
|
link = linkObject.href
|
|
|
|
link = link.replace('{(original)}', data.original)
|
|
|
|
link = link.replace('{(majorVersion)}', '' + (data.majorVersion ? 0))
|
|
|
|
link = link.replace('{($)}', data)
|
2014-02-15 20:29:54 -05:00
|
|
|
@getOrMakeModelFromLink(link, shouldLoadProjection)
|
2014-01-03 13:32:13 -05:00
|
|
|
|
2014-02-15 20:29:54 -05:00
|
|
|
@getOrMakeModelFromLink: (link, shouldLoadProjection=null) ->
|
2014-01-03 13:32:13 -05:00
|
|
|
makeUrlFunc = (url) -> -> url
|
|
|
|
modelUrl = link.split('/')[2]
|
|
|
|
modelModule = _.string.classify(modelUrl)
|
|
|
|
modulePath = "models/#{modelModule}"
|
|
|
|
window.loadedModels ?= {}
|
|
|
|
|
|
|
|
try
|
|
|
|
Model = require modulePath
|
|
|
|
window.loadedModels[modulePath] = Model
|
|
|
|
catch e
|
|
|
|
console.error 'could not load model from link path', link, 'using path', modulePath
|
|
|
|
return
|
|
|
|
|
|
|
|
model = new Model()
|
2014-02-15 20:29:54 -05:00
|
|
|
if shouldLoadProjection? model
|
|
|
|
sep = if link.search(/\?/) is -1 then "?" else "&"
|
|
|
|
link += sep + "project=true"
|
2014-01-03 13:32:13 -05:00
|
|
|
model.url = makeUrlFunc(link)
|
|
|
|
return model
|
|
|
|
|
|
|
|
@isObjectID: (s) ->
|
|
|
|
s.length is 24 and s.match(/[a-z0-9]/gi)?.length is 24
|
|
|
|
|
|
|
|
module.exports = CocoModel
|