codecombat/app/models/SuperModel.coffee

347 lines
12 KiB
CoffeeScript
Raw Normal View History

2014-04-09 15:11:59 -04:00
module.exports = class SuperModel extends Backbone.Model
2014-01-03 13:32:13 -05:00
constructor: ->
@num = 0
@denom = 0
@progress = 0
@resources = {}
@rid = 0
@maxProgress = 1
2014-01-03 13:32:13 -05:00
@models = {}
@collections = {}
# Since the supermodel has undergone some changes into being a loader and a cache interface,
# it's a bit wonky to use. The next couple functions are meant to cover the majority of
# use cases across the site. If they are used, the view will automatically handle errors,
# retries, progress, and filling the cache. Note that the resource it passes back will not
# necessarily have the same model or collection that was passed in, if it was fetched from
# the cache.
report: ->
# Useful for debugging why a SuperModel never finishes loading.
2014-06-30 22:16:26 -04:00
console.info 'SuperModel report ------------------------'
console.info "#{_.values(@resources).length} resources."
unfinished = []
for resource in _.values(@resources) when resource
2016-04-08 15:59:10 -04:00
console.info "\t", resource.name, 'loaded', resource.isLoaded, resource.model
unfinished.push resource unless resource.isLoaded
unfinished
loadModel: (model, name, fetchOptions, value=1) ->
# Deprecating name. Handle if name is not included
value = fetchOptions if _.isNumber(fetchOptions)
fetchOptions = name if _.isObject(name)
# hero-ladder levels need remote opponent_session for latest session data (e.g. code)
# Can't apply to everything since other features rely on cached models being more recent (E.g. level_session)
# E.g.#2 heroConfig isn't necessarily saved to db in world map inventory modal, so we need to load the cached session on level start
cachedModel = @getModelByURL(model.getURL()) unless fetchOptions?.cache is false and name is 'opponent_session'
if cachedModel
if cachedModel.loaded
res = @addModelResource(cachedModel, name, fetchOptions, 0)
res.markLoaded()
return res
else
res = @addModelResource(cachedModel, name, fetchOptions, value)
res.markLoading()
return res
else
@registerModel(model)
res = @addModelResource(model, name, fetchOptions, value)
2014-05-08 14:10:22 -04:00
if model.loaded then res.markLoaded() else res.load()
return res
loadCollection: (collection, name, fetchOptions, value=1) ->
# Deprecating name. Handle if name is not included
value = fetchOptions if _.isNumber(fetchOptions)
fetchOptions = name if _.isObject(name)
url = collection.getURL()
if cachedCollection = @collections[url]
console.debug 'Collection cache hit', url, 'already loaded', cachedCollection.loaded
if cachedCollection.loaded
res = @addModelResource(cachedCollection, name, fetchOptions, 0)
res.markLoaded()
return res
else
res = @addModelResource(cachedCollection, name, fetchOptions, value)
res.markLoading()
return res
else
@addCollection collection
onCollectionSynced = (c) ->
if collection.url is c.url
@registerCollection c
else
console.warn 'Sync triggered for collection', c
console.warn 'Yet got other object', c
@listenToOnce collection, 'sync', onCollectionSynced
@listenToOnce collection, 'sync', onCollectionSynced
res = @addModelResource(collection, name, fetchOptions, value)
res.load() if not (res.isLoading or res.isLoaded)
return res
# Eventually should use only these functions. Use SuperModel just to track progress.
trackModel: (model, value) ->
2016-03-01 13:33:38 -05:00
res = @addModelResource(model, '', {}, value)
res.listen()
2016-02-24 14:40:21 -05:00
trackCollection: (collection, value) ->
res = @addModelResource(collection, '', {}, value)
res.listen()
trackRequest: (jqxhr, value=1) ->
res = new Resource('', value)
res.jqxhr = jqxhr
jqxhr.done -> res.markLoaded()
jqxhr.fail -> res.markFailed()
@storeResource(res, value)
2016-04-13 12:54:24 -04:00
return jqxhr
2016-04-13 12:54:24 -04:00
trackRequests: (jqxhrs, value=1) -> @trackRequest(jqxhr, value) for jqxhr in jqxhrs
2016-02-24 14:40:21 -05:00
# replace or overwrite
shouldSaveBackups: (model) -> false
# Caching logic
2014-01-03 13:32:13 -05:00
getModel: (ModelClass_or_url, id) ->
return @getModelByURL(ModelClass_or_url) if _.isString(ModelClass_or_url)
m = new ModelClass_or_url(_id: id)
return @getModelByURL(m.getURL())
2014-01-03 13:32:13 -05:00
getModelByURL: (modelURL) ->
modelURL = modelURL() if _.isFunction(modelURL)
2014-01-03 13:32:13 -05:00
return @models[modelURL] or null
2014-08-30 12:19:41 -04:00
getModelByOriginal: (ModelClass, original, filter=null) ->
_.find @models, (m) ->
m.get('original') is original and m.constructor.className is ModelClass.className and (not filter or filter(m))
2014-01-03 13:32:13 -05:00
getModelByOriginalAndMajorVersion: (ModelClass, original, majorVersion=0) ->
_.find @models, (m) ->
return unless v = m.get('version')
m.get('original') is original and v.major is majorVersion and m.constructor.className is ModelClass.className
2014-01-03 13:32:13 -05:00
getModels: (ModelClass) ->
# can't use instanceof. SuperModel gets passed between windows, and one window
# will have different class objects than another window.
# So compare className instead.
return (m for key, m of @models when m.constructor.className is ModelClass.className) if ModelClass
return _.values @models
registerModel: (model) ->
@models[model.getURL()] = model
2014-01-03 13:32:13 -05:00
getCollection: (collection) ->
return @collections[collection.getURL()] or collection
2014-01-03 13:32:13 -05:00
addCollection: (collection) ->
# TODO: remove, instead just use registerCollection?
url = collection.getURL()
if @collections[url]? and @collections[url] isnt collection
2014-01-03 13:32:13 -05:00
return console.warn "Tried to add Collection '#{url}' to SuperModel when we already had it."
@registerCollection(collection)
2014-01-03 13:32:13 -05:00
registerCollection: (collection) ->
@collections[collection.getURL()] = collection if collection.isCachable
2014-01-03 13:32:13 -05:00
# consolidate models
for model, i in collection.models
cachedModel = @getModelByURL(model.getURL())
2014-01-03 13:32:13 -05:00
if cachedModel
clone = $.extend true, {}, model.attributes
cachedModel.set(clone, {silent: true, fromMerge: true})
#console.debug "Updated cached model <#{cachedModel.get('name') or cachedModel.getURL()}> with new data"
2014-01-03 13:32:13 -05:00
else
@registerModel(model)
2014-01-03 13:32:13 -05:00
collection
# Tracking resources being loaded for this supermodel
2014-01-03 13:32:13 -05:00
finished: ->
2016-04-08 15:59:10 -04:00
return (@progress is 1.0) or (not @denom) or @failed
2014-04-09 15:11:59 -04:00
addModelResource: (modelOrCollection, name, fetchOptions, value=1) ->
# Deprecating name. Handle if name is not included
value = fetchOptions if _.isNumber(fetchOptions)
fetchOptions = name if _.isObject(name)
modelOrCollection.saveBackups = modelOrCollection.saveBackups or @shouldSaveBackups(modelOrCollection)
2014-04-09 15:11:59 -04:00
@checkName(name)
res = new ModelResource(modelOrCollection, name, fetchOptions, value)
@storeResource(res, value)
2014-04-09 15:11:59 -04:00
return res
removeModelResource: (modelOrCollection) ->
@removeResource _.find(@resources, (resource) -> resource?.model is modelOrCollection)
addRequestResource: (name, jqxhrOptions, value=1) ->
# Deprecating name. Handle if name is not included
value = jqxhrOptions if _.isNumber(jqxhrOptions)
jqxhrOptions = name if _.isObject(name)
2014-04-09 15:11:59 -04:00
@checkName(name)
res = new RequestResource(name, jqxhrOptions, value)
@storeResource(res, value)
2014-04-09 15:11:59 -04:00
return res
addSomethingResource: (name, value=1) ->
value = name if _.isNumber(name)
2014-04-09 15:11:59 -04:00
@checkName(name)
res = new SomethingResource(name, value)
@storeResource(res, value)
2014-04-09 15:11:59 -04:00
return res
checkName: (name) ->
#if _.isString(name)
# console.warn("SuperModel name property deprecated. Remove '#{name}' from code.")
2014-04-09 15:11:59 -04:00
storeResource: (resource, value) ->
@rid++
resource.rid = @rid
@resources[@rid] = resource
@listenToOnce(resource, 'loaded', @onResourceLoaded)
@listenTo(resource, 'failed', @onResourceFailed)
@denom += value
_.defer @updateProgress if @denom
2014-04-09 15:11:59 -04:00
removeResource: (resource) ->
return unless @resources[resource.rid]
@resources[resource.rid] = null
--@num if resource.isLoaded
--@denom
_.defer @updateProgress
onResourceLoaded: (r) ->
return unless @resources[r.rid]
@num += r.value
_.defer @updateProgress
r.clean()
@stopListening r, 'failed', @onResourceFailed
@trigger 'resource-loaded', r
2014-04-09 15:11:59 -04:00
onResourceFailed: (r) ->
return unless @resources[r.rid]
@failed = true
@trigger('failed', resource: r)
r.clean()
2014-04-09 15:11:59 -04:00
updateProgress: =>
# Because this is _.defer'd, this might end up getting called after
# a bunch of things load all at once.
# So make sure we only emit events if @progress has changed.
newProg = if @denom then @num / @denom else 1
newProg = Math.min @maxProgress, newProg
return if @progress >= newProg
@progress = newProg
@trigger('update-progress', @progress)
@trigger('loaded-all') if @finished()
setMaxProgress: (@maxProgress) ->
resetProgress: -> @progress = 0
clearMaxProgress: ->
@maxProgress = 1
_.defer @updateProgress
getProgress: -> return @progress
2014-04-09 15:11:59 -04:00
getResource: (rid) ->
return @resources[rid]
# Promises
finishLoading: ->
new Promise (resolve, reject) =>
return resolve(@) if @finished()
@once 'failed', ({resource}) ->
jqxhr = resource.jqxhr
reject({message: jqxhr.responseJSON?.message or jqxhr.responseText or 'Unknown Error'})
@once 'loaded-all', => resolve(@)
2014-04-09 15:11:59 -04:00
class Resource extends Backbone.Model
constructor: (name, value=1) ->
2014-04-09 15:11:59 -04:00
@name = name
@value = value
@rid = -1 # Used for checking state and reloading
2014-04-09 15:11:59 -04:00
@isLoading = false
@isLoaded = false
@model = null
@jqxhr = null
2014-04-09 15:11:59 -04:00
markLoaded: ->
return if @isLoaded
@trigger('loaded', @)
2014-04-09 15:11:59 -04:00
@isLoaded = true
@isLoading = false
markFailed: ->
return if @isLoaded
@trigger('failed', @)
@isLoaded = @isLoading = false
@isFailed = true
markLoading: ->
@isLoaded = @isFailed = false
@isLoading = true
clean: ->
# request objects get rather large. Clean them up after the request is finished.
@jqxhr = null
load: -> @
2014-04-09 15:11:59 -04:00
class ModelResource extends Resource
constructor: (modelOrCollection, name, fetchOptions, value)->
super(name, value)
@model = modelOrCollection
@fetchOptions = fetchOptions
@jqxhr = @model.jqxhr
2016-08-02 15:01:12 -04:00
@loadsAttempted = 0
2014-04-09 15:11:59 -04:00
load: ->
2016-08-02 15:01:12 -04:00
timeToWait = 5000
tryLoad = =>
return if this.isLoaded
if @loadsAttempted > 4
@markFailed()
return @
@markLoading()
if @loadsAttempted > 0
console.log "Didn't load model in #{timeToWait}ms (attempt ##{@loadsAttempted}), trying again: ", this
@fetchModel()
@listenTo @model, 'error', (levelComponent, request) ->
if request.status not in [408, 504, 522, 524]
2016-08-02 15:01:12 -04:00
clearTimeout(@timeoutID)
clearTimeout(@timeoutID) if @timeoutID
@timeoutID = setTimeout(tryLoad, timeToWait)
@loadsAttempted += 1
timeToWait *= 1.5
tryLoad()
@
2014-04-09 15:11:59 -04:00
fetchModel: ->
@jqxhr = @model.fetch(@fetchOptions) unless @model.loading
@listen()
2016-02-24 14:40:21 -05:00
listen: ->
@listenToOnce @model, 'sync', -> @markLoaded()
@listenToOnce @model, 'error', -> @markFailed()
clean: ->
@jqxhr = null
@model.jqxhr = null
2014-04-09 15:11:59 -04:00
class RequestResource extends Resource
constructor: (name, jqxhrOptions, value) ->
2014-04-09 15:11:59 -04:00
super(name, value)
@jqxhrOptions = jqxhrOptions
load: ->
@markLoading()
@jqxhr = $.ajax(@jqxhrOptions)
# make sure any other success/fail callbacks happen before resource loaded callbacks
@jqxhr.done => _.defer => @markLoaded()
@jqxhr.fail => _.defer => @markFailed()
@
2014-04-09 15:11:59 -04:00
class SomethingResource extends Resource