Research
Security News
Malicious npm Packages Inject SSH Backdoors via Typosquatted Libraries
Socket’s threat research team has detected six malicious npm packages typosquatting popular libraries to insert SSH backdoors.
Classy offers the ability to easily define classes, call super or overriden methods, define static properties, and mixin objects in a very flexible way.
Meant to be used in the browser and in node.
For the browser, use dist/classy.js
var Vehicle = classy.define({
alias: 'vehicle',
init: function(year){
this.year = year
}
})
var Car = classy.define({
extend: 'vehicle'
//or extend: Vechicle
alias: 'car',
init: function(year, make){
this.callSuper()
this.make = make
},
getName: function(){
return this.make
}
})
var ford = new Car(1980, 'Ford')
console.log(ford.year)
console.log(ford.make)
Notice the callSuper()
method call, which can be used in any class method, and will call the method with the same name found on the super class. It also automatically transmits all the arguments it has, so you don't have to manually do so.
ford.getName() === 'Ford' //is true
classy.override('car', {
getName: function(){
return this.callOverriden() + ', made in ' + this.year
}
})
//now
ford.getName() === 'Ford, made in 1980' //is true
You can use the class alias
in order to easily reference which class you want to extend or override. This also helps you get a reference to your class by
var Car = classy.getClass('car')
var Vehicle = classy.getClass('vehicle')
Overriding is simple, just call classy.override
classy.override(Car, {
getName: function(){
return this.callOverriden() + ', great car'
}
})
or, if you don't have a reference to the class, but only have the alias
classy.override('car', {
getName: function(){
return this.callOverriden() + ', great car'
}
})
init
as constructorUse the init
method as the constructor
Example
var Animal = classy.define({
//when a new Animal is created, the init method is called
init: function(config){
config = config || {}
//we simply copy all the keys onto this
Object.keys(config).forEach(function(key){
this[key] = config[key]
}, this)
}
})
var Cat = classy.define({
extend: Animal,
alias: 'cat',
init: function(){
this.callSuper()
this.sound = 'meow'
},
getName: function(){
return this.name
}
})
var lizzy = new Cat({ name: 'lizzy' })
callSuper
and callOverriden
Use the callSuper
and callOverriden
methods to call the super and overriden methods. You don't have to worry about forwarding the arguments, since this is handled automagically for you.
If there is no super or overriden method with the same name you don't have to worry either, since callSuper and callOverriden won't break. they will simply and silently do nothing
Example
//create a shape class
classy.define({
alias: 'shape',
getDescription: function(){
return this.name
}
})
//create a rectangle class with a width and a height
classy.define({
extend: 'shape',
alias: 'rectangle',
name: 'rectangle',
init: function(size){
this.width = size.width
this.height = size.height
},
getArea: function(){
return this.width * this.height
},
setHeight: function(h){ this.height = h },
setWidth: function(w){ this.width = w }
})
classy.override('rectangle', {
getDescription: function(){
//reimplement the getDescription, but use the overriden implementation as well
return 'this is a ' + this.callOverriden()
}
})
//create a square class
classy.define({
extend: 'rectangle',
alias: 'square',
init: function(size){
if (size * 1 == size){
//the size is a number
size = { width: size, height: size}
} else {
size.width = size.height
}
this.callSuper()
},
setHeight: function(h){
//callSuper will automatically pass the arguments to Rectangle.setHeight, so h will be forwarded
this.callSuper() //or you could use this.callSuperWith(10) if you want to manually pass parameters
this.setWidth(h)
}
})
You can also use callSuperWith
and callOverridenWith
to manually pass all parameters
Example
//...
setHeight: function(h){
this.callSuperWith(h*2)
}
//...
Classy offers the ability to mix objects into other objects. At a base level, you can either use simple objects as mixins or you can define mixin classes.
Example
var logger = {
$after: {
log: function(msg){
console.log(msg)
}
}
}
var person = { firstName: 'Bob', lastName: 'Johnson' }
classy.mixin(person /* target object */, logger /* mixin */)
in the example above, the person object receives a log function property. Note the usage of $after. Other valid mixin behaviors are $copyIf
, $before
and $override
.
These behaviors determine how mixin properties that are functions are mixed-in when the target object already has those properties.
Any property in the mixin is copied onto the target object only if the target object does not already have a property with the same name
Example
var logger = {
$copyIf: {
isLogger: true,
log: function(msg){ console.log(msg) },
greet: function(msg){ console.log('hello ' + msg) }
}
}
var person = {
greet: function(msg){
alert('Hi ' + msg)
}
}
classy.mixin(person, logger)
person.greet('Bob') //will alert 'Hi Bob' - so logger.greet is not copied, since it already existed on person
person.log('warning') //will console.log 'warning' - logger.log was copied to person, since person.log was undefined
person.isLogger === true
classy.defineMixin({
alias: 'logger',
$before: {
log: function(msg){
console.log(msg)
},
warn: function(warning){
console.warn(warning)
return '!'
}
}
})
var person = {
log: function(msg){ alert(msg); return 1}
}
classy.mixin(person, 'logger')
person.log('hi') === 1 // will first console.log('hi') and then will alert('hi')
//and will return the return value of the initial person.log implementation
person.warn('hi') === '!' //will console.warn('hi')
In the above example, since log and warn ar copied with a before behavior, first of all classy checks to see if person already has those properties. Since person.log exists, person.log is assigned another function, which calls logger.log
before person.log
, and returns the result of the initial person.log
.
For logger.warn, no such property exists on person, so it is simply assigned to the person.
The behavior of after is similar, with the difference that the mixin function is called after the initial implementation. The result is that of the initial implementation, if one exists.
classy.defineMixin({
alias: 'logger',
$override: {
log: function(msg){ console.log(msg) },
warn: function(msg){
console.log(msg)
return this.callTarget() //call the target object warn implementation, if one exists
}
}
})
var Person = classy.define({
alias: 'person',
mixins: [
'logger'
],
name: 'bob',
warn: function(msg){
alert(msg)
return this
}
})
var p = new Person()
p.log(p.name) //simply calls logger.log
p.warn(p.name) // logs p.name and then alerts p.name
Notice that logger.warn
calls this.callTarget()
which means the mixin tries to call the method from the target object that this function has overriden. Since person.warn had an implementation, the logger calls that.
$ownClass
You can easily define static properties for classes.
var Widget = classy.define({
statics: {
idSeed: 0,
getDescription: function(){
return 'A Widget class'
},
getNextId: function(){
return this.idSeed++
}
},
init: function(){
this.id = this.$ownClass.getNextId()
}
})
Widget.getDescription() == 'A Widget class' // === true
var w = new Widget()
w.id === 0
w = new Widget()
w.id === 1
On every instance, you can use the $ownClass property in order to get a reference to the class that created the instance.
FAQs
Classy - Classes for JavaScript
The npm package classy receives a total of 16 weekly downloads. As such, classy popularity was classified as not popular.
We found that classy demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 1 open source maintainer collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Research
Security News
Socket’s threat research team has detected six malicious npm packages typosquatting popular libraries to insert SSH backdoors.
Security News
MITRE's 2024 CWE Top 25 highlights critical software vulnerabilities like XSS, SQL Injection, and CSRF, reflecting shifts due to a refined ranking methodology.
Security News
In this segment of the Risky Business podcast, Feross Aboukhadijeh and Patrick Gray discuss the challenges of tracking malware discovered in open source softare.