This file is part of Heterarchy.
- View me on a static web
- View me on GitHub
Adds multiple inheritance support to CoffeeScript (and JavaScript). It uses the C3 linearization algorithm as described in the famous Dylan paper.
Like Underscore.js the module can be used both with Node.js as well as in the browser. Therefore, global variables are set accordingly.
# node.js
if typeof global is "object" and global?.global is global
root = global
exports = module.exports
_ = require "underscore"
# browser
else
root = window
exports = window.heterarchy = {}
_ = window._
Underscore.js
is used to save lots of common-problem code and
assert
is used upon an invalid inheritance hierarchy.
{head, tail, map, find, some, without, isEmpty, every, memoize, reject,
isEqual, reduce} = _
assert = (value, errorMessage) ->
if not value
throw new Error(errorMessage)
The multi function takes a list of classes and returns a special class object that merges the class hierarchies, as linearized by the C3 algorithm. Limitations of the approach are:
-
instanceof
does not always work as expected. For example:class A class B extends A class C extends A class D extends multi B, C assert new D not instanceof B
Instead, one should use the provided
isinstance
function. -
Some of the bases of a multi-inherited hierarchy are frozen when the sub-class is defined -- i.e. later modifications to the superclass are not visible to the subclass or its instances. For example, in the previous heterarchy:
B::newProperty = 42 assert D::newProperty == undefined
The multi
function memoizes its results such that identity is
maintained, implying multi X, Y is multi X, Y
.
exports.multi = (bases...) ->
cls = generate merge map(bases, mro).concat [bases]
cls.__bases__ = bases
cls
This takes a list of classes representing a hierarchy (from most to least derived) and generates a single-inheritance hierarchy that behaves like a class that would have such a hierarchy.
generate = memoize (linearization) ->
next = head linearization
if isEqual linearization, hierarchy next
next
else
class Result extends generate tail linearization
__mro__: linearization
constructor: reparent next, @, next::constructor
# 1. Fill up missing class attributes and
# 2. Adjust class methods (so the MRO is used).
# Those are already part of the class because when extending
# all class attributes are copied to `this` by CoffeeScript.
for own key, value of next
@[key] = reparent next, @, value
# fill up missing instance attributes
for own key, value of next::
if not @::hasOwnProperty key
@::[key] = reparent next, @, value
Methods in CoffeeScript call super directly, so we have to change the
__super__
attribute of the original class during the scope of the
method so it calls the right super of the linearization. Also,
programmers don't call super in constructor of root classes --indeed
doing so would rise an error-- so we have to inject such a call when
there are classes after these in the linearization. The reparent
function takes care of all these and given an original class, and the
new class that is replacing it a linearized heterarchy, returns a
wrapped copy of a value of the former that is suitable for replacing
it in the later.
reparent = (oldklass, newklass, value) ->
if value not instanceof Function
value
else if value is oldklass::constructor and inherited(oldklass) is Object
superctor = inherited(newklass)::constructor
->
superctor.apply @, arguments
value.apply @, arguments
else
newsuper = inherited(newklass)::
oldsuper = oldklass.__super__
->
oldklass.__super__ = newsuper
try
value.apply @, arguments
finally
oldklass.__super__ = oldsuper
This is the C3 linearization algorithm, as translated from the original paper.
merge = (inputs) ->
while not isEmpty inputs
next = find (map inputs, head), (candidate) ->
every inputs, (input) -> candidate not in tail input
assert next?, "Inconsistent multiple inheritance"
inputs = reject map(inputs, (lst) -> without lst, next), isEmpty
next
The mro function returns the method resolution order (linearization) of a given class:
class A class B extends A class C extends B assert mro(C).equals [C, B, A]
It returns the original classes that were mixed in when used with multi-inherited classes:
class A class B extends A class C extends A class D extends multi B, C assert mro(D).equals [D, B, C, A, Object]
javaScriptClassNames = [
"Array"
"Boolean"
"Date"
"Error"
"Function"
"Number"
"RegExp"
"String"
"Object"
"EvalError"
"RangeError"
"ReferenceError"
"SyntaxError"
"TypeError"
"URIError"
# non-standard classes
"Symbol"
# typed arrays
"Int8Array"
"Uint8Array"
"Uint8ClampedArray"
"Int16Array"
"Uint16Array"
"Int32Array"
"Uint32Array"
"Float32Array"
"Float64Array"
# Keyed collections
"Map"
"Set"
"WeakMap"
"WeakSet"
# Structured data
"ArrayBuffer"
"DataView"
# Control abstraction objects
"Promise"
"Generator"
"GeneratorFunction"
# Reflection
"Reflect"
"Proxy"
]
javaScriptClasses = reduce javaScriptClassNames, (classes, name) ->
classes[root[name]] = root[name]
classes
, {}
isJavaScriptClass = (cls) ->
javaScriptClasses[cls] is cls
exports.mro = mro = (cls) ->
if not cls? or not cls::?
[]
else if not cls::hasOwnProperty "__mro__"
result = [cls].concat mro inherited(cls)
cls::__mro__ = result unless isJavaScriptClass cls
result
else
cls::__mro__
The bases function returns an array of the base- / superclasses of a given class. This works for single and multiple inheritance and classes without a superclass.
exports.bases = (cls) ->
# heterarchy's multiple inheritance (see `multi`)
if cls?.__bases__?
cls.__bases__
# CoffeeScript's single inheritance
else if cls?.__super__?
[cls.__super__.constructor]
else
[]
The inherited function returns the CoffeeScript superclass of an object, for example:
class A class B extends A assert inherited(B) == A
Note that for multiple inherited classes, this returns the mixed object, not the next class in the MRO, as in:
class C extends multi A, B assert inherited(C) == multi(A, B)
exports.inherited = inherited = (cls) ->
Object.getPrototypeOf(cls.prototype)?.constructor
The hierarchy returns the CoffeeScript hierarchy of classes of a given class, including the class itself. For multiple inherited classes, it may return special classes that were generated to produce the flattening, as in:
class A class B extends A class C extends A class D extends multi B, C assert not mro(D).equals hierarchy(D) assert hierarchy(D).equals [ D, multi(B, C), inherited(multi B, C), A, Object ]
exports.hierarchy = hierarchy = (cls) ->
if not cls?
[]
else
[cls].concat hierarchy inherited cls
The isinstance function takes an object and a class or classes and returns whether the object is an instance of any of those classes. It is compatible with multi-inherited classes.
exports.isinstance = (obj, classes...) ->
exports.issubclass obj?.constructor, classes...
The issubclass tells whether a class is a subtype of another type.
exports.issubclass = (klass, classes...) ->
linearization = mro klass
some classes, (cls) -> cls in linearization
Copyright (c) 2013, 2015 Juan Pedro Bolivar Puente [email protected]
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.