Security News
Fluent Assertions Faces Backlash After Abandoning Open Source Licensing
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
candy-wrapper
Advanced tools
Wrap your functions, methods, and properties to monitor them and modify their behavior.
This project is currently a RELEASE CANDIDATE. Please provide feedback on bugs or improvements before the 1.0 release.
Full API Documentation Available Here!
Questions, comments, bugs, suggestions, contributions, etc. are welcomed. Please submit an issue on GitHub.
This library provides the ability to wrap functions or properties so that you can monitor their behavior without changing it. Wrappers also allow you to decide when and how you want to change the behavior of a function or property. Candy-wrapper is similar to wonderful libraries like Sinon and shimmer and can be used for creating stubs, spys, and mocks for testing and monkey patching in production environments. The motivation behind creating candy-wrapper was that other libraries were too hard to learn, had inconsistent APIs, or generally couldn't do the things I needed to do with a wrapper. Hopefully candy-wrapper does everything anyone could ever hope for from a wrapper, or can be easily expanded to add new functionality.
The primary interfaces are Wrapper, Expect, Trigger, and Action:
If you are new to wrappers or candy-wrapper, a good place to start is the Getting Started Tutorial.
There is also a higher-level interface for defining stubs and tests based on their behavior. This interface is especially useful for creating stubs and tests that mirror each other, where the stubs can be used to replace a module during testing and a test can make sure that the module has the same behavior as a stub. The primary interface for this functionality is:
require
. After behaviors are defined for the module, stubs and tests can automatically be generated to stub out the module where it is required or test instances of the module.For a quick introduction to Modules, Intefaces and Behaviors, see the Modules and Behaviors Tutorial
Install:
$ npm install candy-wrapper
Use:
var Wrapper = require("candy-wrapper").Wrapper;
new Wrapper();
Install:
<script src=https://rawgit.com/apowers313/candy-wrapper/master/candy-wrapper.js></script>
Use:
<script>
var Wrapper = CandyWrapper.Wrapper;
</script>
Here are few example use cases to get you started with candy-wrapper...
Let's say that you have already written some code and now you want to test it to make sure it works right. Your very simple function square()
simply takes a number and squares it and you want to make sure it's working right.
// our simple square function, squares the number passed in
// throws error if the argument isn't a number
var square = function(num) {
if (typeof num !== "number") {
throw new TypeError ("expected argument to be Number");
}
return num * num;
};
// wrap square so that we can analyze it later
square = new Wrapper(square);
// let's make some calls to square
square(2);
square(4);
try { // gobble up the exception from no arguments
square();
} catch(e) {}
square(3);
square(5);
square.historyList.filterFirst().expectReturn(4); // true
square.historyList.filterSecond().expectReturn(16); // true
square.historyList.filterThird().expectException(new TypeError ("expected argument to be Number")); // true
square.historyList.filterFourth().expectReturn(10); // false, actually returned 9
square.historyList.filterFifth().expectReturn(23); // false, actually returned 25
square.expectReportAllFailures();
// ExpectError: 2 expectation(s) failed:
// expectReturn: expectation failed for: 10
// expectReturn: expectation failed for: 23
Let's say you're working on a big complex system that's eventually going to hook up to a database, but you really don't want to have to write all the database code before you can start working on the more interesting code that would rely on it. If only you could create a fake database object that behaved like the real thing...
var fakeDb = {};
fakeDb.getUser = new Wrapper(); // create a stub function for getUser...
fakeDb.listUsers = new Wrapper(); // create a stub function for listUsers...
// when `fakeDb.getUser` is called with the argument "apwoers", it returns the object below
fakeDb.getUser.triggerOnArgs("apowers")
.actionReturn({
username: "apowers",
firstName: "Adam",
lastName: "Powers"
});
// everytime `fakeDb.listUsers` is called, it returns the same array...
fakeDb.listUsers.triggerAlways()
.actionReturn([{
username: "apowers",
firstName: "Adam",
lastName: "Powers"
}, {
username: "bhope",
firstName: "Bob",
lastName: "Hope"
}]);
var user = fakeDb.getUser("apowers"); // returns a user object...
var userList = fakeDb.listUsers(); // returns an array of users...
Now let's say it's been a few months, and your system is almost completely done. You have completed your real database, and you're having some sort of error that you suspect is due to your database timing out. You want to simulate errors with your database, but it's not easy to figure out how to make it time out when you want it to. Instead, you can just wrap your database object in a mock and start injecting errors...
// this is your big complex interface to a MongoDB database
var mysqlDbInterface = {
// ...
getUser: function() {
// make database call here
}
// ...
};
var getUserWrapper = new Wrapper(mysqlDbInterface, "getUser");
getUserWrapper.triggerOnCallNumber(2)
.actionThrowException(new Error("Connection to mySQL timed out"));
// start server
// the third call to `mysqlDbInterface.getUser()` will throw a timeout Error now..
Let's say that you are tired of Math.random() being a pseudo-random number generator and you want it to use your Digital Random Number Generator (DRNG) instead. Using candy-wrapper, you can monkey patch (that is, replace) Math.random()
with your DRNG:
// replace "Math.random" with the new function of our choosing...
new Wrapper(Math, "random", function() {
return 1.2345; // this is where you would call your DRNG...
});
// let's see what it does...
Math.random() // returns 1.2345
// check to see if Math.random is a Wrapper
Wrapper.isWrapper(Math, "random"); // true
// this is probably a bad idea, let's go back to the original random number generator...
Wrapper.unwrap(Math, "random");
Changing gears (so to speak), our next example shows how to define a new module that describes the functionality of a car. The {@link Module} can then create a stub to be used where the module is needed during testing, and tests to ensure that any instance of the module will behave properly when imported into the real system.
// create a new module that will define a car
var carModule = new Module();
// define a function for the module
carModule.defineMethod("startEngine");
// successfully start the engine
carModule.defineBehavior("startEngineSuccess")
.startEngine()
.args("key")
.returns(true);
// fail to start the engine because no key was provided
carModule.defineBehavior("startEngineNoKey")
.startEngine()
.args()
.throws(new Error("no key provided"));
// define a property for the module
carModule.defineProperty("currentSpeed");
// property returns 10 when going slow
carModule.defineBehavior("currentSpeedSlow")
.currentSpeed()
.returns(10);
// property returns 70 when at cruising speed
carModule.defineBehavior("currentSpeedCruise")
.currentSpeed()
.returns(70);
// property returns 120 when going fast
carModule.defineBehavior("currentSpeedFast")
.currentSpeed()
.returns(120);
// define a more complex behavior based on the behaviors we already defined
carModule.defineBehavior("accelerate")
.startEngineSuccess()
.currentSpeedSlow()
.currentSpeedCruise()
.currentSpeedFast();
/*****************************************************
* A stub that replaces the functionality for testing
/****************************************************/
// normally this is what happens
var carModule = require("carModule");
// create a stub that mocks the `startEngineSuccess` behavior
// maybe replace the normal module with your stub using something like Mockery:
// https://www.npmjs.com/package/mockery
var carModule = carModule.getStub("startEngineSuccess");
/*****************************************************
* A module that will have it's behavior tested
/****************************************************/
// define which behaviors should be tested for the module
carModule.defineTest("startEngineSuccess", "start the car's engine");
carModule.defineTest("startEngineNoKey", "fails to start the engine without a key");
carModule.defineTest("accelerate", "start the engine and current speed gets faster");
// your instance of the car module
var myFerrari = {
startEngine = function(key) {
if (key !== "key") throw new Error ("no key provided");
return true;
},
currentSpeed: 0
};
// maybe your module would do something like this
module.exports = myFerrari;
// run all the tests against your module using the 'describe' and 'it' functions from Mocha
describe ("my ferrari", function() {
mod.runAllTests(myFerrari, it);
});
Note that this library currently makes exensive use of the features of JavaScript ES6, notably Proxies, classes, template literals, rest parameters, and the spread operator. For information about which platforms currently support ES6, see the ES6 compatibility table. It has been tested against Node 6 and is tested against the latest versions of Chrome, Firefox, Edge, and Safari.
Presumably it could be back-ported to ES5.1, perhaps drawing on the work of Sinon. The biggest challenge would be removing the use of Proxies. A pull request would be more than welcome if anyone wants to take that challenge on.
FAQs
Wrap your functions, methods, and properties to monitor them and modify their behavior.
We found that candy-wrapper 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.
Security News
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
Research
Security News
Socket researchers uncover the risks of a malicious Python package targeting Discord developers.
Security News
The UK is proposing a bold ban on ransomware payments by public entities to disrupt cybercrime, protect critical services, and lead global cybersecurity efforts.