Skip to content

Curry-like secret sauce for option-object accepting function

License

Notifications You must be signed in to change notification settings

imbcmdth/masala

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

46 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

MASALA

Mix the secret sauce of curry-like functionality into your function's option-objects without anything too spicy

browser support

Contents

Install

npm install masala

..then require masala:

var masala = require('masala');

In the browser, traditional

For the browser, add the following to your pages:

<script src="masala.js"></script>

And the global function masala will be available.

In the browser, using AMD (require.js)

...Or using AMD in the browser:

require(["masala"], function(masala) {
	// ...
});

Basic Usage

function addAB (options) {
	return options.a + options.b;
}

//-- Creating a masala'd function is pretty straight
//-- forward:
var add = masala(addAB, { a: undefined, b: undefined });

//-- Simply pass an object that serves as a set of default
//-- options. Properties set to undefined become *required* options
//-- that must be provided (and not undefined) before the function
//-- is evaluated.

//-- It can be called like normal:
add({ a: 1, b: 2 }) //=> 3

//-- Or, if you leave out any arguments,
//-- a new function that expects all (or some) of
//-- the remaining arguments will be created:
var add1 = add({ a: 1 });
add1({ b: 2 }) //=> 3

//-- Already provided options can be overridden at any time:
add1({ a: 2, b: 2 }) //=> 4
//-- ..giving you default options behavior without adding clutter
//-- to your functions.

//-- Masala knows how many options a function should take
//-- by the number of `undefined` parameters in the defaults object

//-- In this case, the function expects an object with two arrays is
//-- expected (a, b).
//-- `zipWith` will combine two arrays using a function (fn):
var zipWith = masala(function (opts) {
	return opts.a.map(function (val, i) { return opts.fn({ a: val, b: opts.b[i] }) });
}, {
	fn: undefined,
	a: undefined,
	b: undefined
});

//-- If there are still more arguments required, a masala'd function
//-- will always return a new masala'd function:
var zipAdd = zipWith({ fn: add });
var zipAddWith123 = zipAdd({ a: [1, 2, 3] });

//-- Both functions are usable as you'd expect at any time:
zipAdd({ a: [1, 2, 3], b: [1, 2, 3] }); //=> [2, 4, 6]
zipAddWith123({ b: [5, 6, 7] }) //=> [6, 8, 10]

Intermediate Usage

//-- Masala also acts like curry for any remaining arguments after the options

function multiplyDivide (options, denom) {
	return (options.a * options.b) / denom;
}

var multDiv = masala(multiplyDivide, { a: undefined, b: undefined });

//-- If the first argument isn't an object, it is assumed to be one of
//-- the remaining arguments:
var multDivBy2 = multDiv(2);
multDivBy2({ a: 3, b: 5 }) //=> 7.5

//-- Otherwise, extra arguments after the first object are used like
//-- you would expect from standard curry-flavoring:
var multBy2Div3 = multDiv({ a: 2 }, 3);
multBy2Div3({ b: 6 }) //=> 4

Advanced Usage

//-- The second argument passed to masala can be a number representing
//-- the position of the options object in the function's parameters
//-- allowing you to curry functions where the options aren't the very
//-- first argument.

function chooseAB (a, options, b) {
	if (options.choice === "a")
		return a;
	else
		return b;
}

var chooser = masala(chooseAB, 1, { choice: undefined });

//-- You can curry the options object and create hard-wired choice:
var chooseA = chooser({ choice: 'a' });
var chooseB = chooser({ choice: 'b' });

chooseA(1, 2) //=> 1
chooseB(1, 2) //=> 2

//-- Or you can apply the arguments and leave the options object pending:
var choose12 = chooser(1, 2);

choose12({ choice: 'a' }) //=> 1
choose12({ choice: 'b' }) //=> 2

//-- Once masala'd the options object is always first and the remaining
//-- arguments are always applied in order:
chooser({choice: 'a'}, 'foo', 'bar') //=> 'foo'

//-- No matter how they are passed:
chooser({choice: 'b'})('foo', 'bar') //=> 'bar'
chooser('foo')({choice: 'a'})('bar') //=> 'foo'

Constructor Usage

function summer (o) {
	this.a = o.a;
	this.b = o.b;
}

summer.prototype.result = function (c) {
	return this.a + this.b + c
};

//-- By using `new` with masala, we tell it that the first argument is
//-- a constructor
var summer = new masala(summer, {
	a: undefined,
	b: undefined
});

//-- Once all the options for the constructor has been supplied, it executes
//-- and returns the constructed object as you would expect.
var sum12 = summer({a: 1, b: 2});

sum12.result(3); //=> 6

Masala also supports something very similar to node.js's util.inherits:

//-- We continue using the same `summer` constructor from above

function sumMore (o) {
	// Call super constructor
	sumMore.super_.apply(this, arguments);
	this.c = o.c;
}

//-- normally we would export this constructor here:
// module.exports = masala.inherits(sumMore, summer, { c: undefined });

//-- But for the purposes of this README we will just assign it to a variable
var sum = masala.inherits(sumMore, summer, { c: undefined });

//-- Now we add any prototype functions:
sumMore.prototype.result = function (d) {
	return sumMore.super_.prototype.result.call(this, d) + this.c;
};

//-- Notice how the `sum` function's options have been merged with it's
//-- superConstructor's options:
sum.options //=> ['a', 'b', 'c']

var temp = sum({a: 1, c: 3});

//-- The `temp` variable isn't an object since not all of the required options
//-- have been bound - it is still waiting for `b`.

var finalSumMore = temp({b: 2});

//-- Now it has all of it's required options so the final object has been
//-- constructed `finalSumMore` is that object.

finalSumMore.result(4); //=> 10

API

[new] masala( yourFunction [[, paramPosition], defaultOptions] )

  • yourFunction function The function to which you wish to add some secret sauce.

  • paramPosition number [optional] The position in yourFunction's parameter list of the options object argument.

  • defaultOptions object Any keys set to undefined become required parameters for the options-currying and any other parameters become default options. Default options can always be overridden later.

If neither paramPosition nor defaultOptions is provided, then masala functions exactly like a traditional curry over all of yourFunction's arguments.

If masala is called as a constructor (ie. with new) then it will invoke the first argument as a constructor function. In this way we can use masala to curry constructors.

newConstructor = masala.inherits( yourConstructor, superConstructor [[, paramPosition], defaultOptions] )

The options are the same as the base masala function's options. The only difference is that this only works with constructors and requires the superConstructor option (currently does no sanity checking so make sure it's included.)

The returned newConstructor is the masala'd constructor that should be exported with module.exports.

The newConstructor inherits any remaining required options from it's superConstructor so that calling the superConstructor in the constructor fucntion will only happen once all it's required options have been specified.

This is probably super confusing so just see the examples.

What are they saying about Masala?

dwcook: "looks neat"

ashnur: "guess it has its usage"

gkatsev: "I give no quotes. Except for money."

hughfdjackson: "this is pretty cool"

Versions

  • v2.0.1 Removed support for null for specifying an unbound option. Now you MUST use only undefined and null is considered a proper, bindable, value

  • v1.3.0 Implemented masala.inherits as a replacement for util.inherits that enables building hierarchies of masala'd functions

  • v1.2.3 Fixed a bug that was overwriting default options under certain circumstances

  • v1.2.2 Added constructor currying with the new operator

  • v1.2.1 Fixed a few small bugs on IE and Safari

  • v1.2.0 The curried function now correctly tracks the arity of remaining arguments and makes available the remaining options-object keys in function.options. In other words, the number of arguments and options remaining before the original function will be executed is function.length + function.options.length.

  • v1.1.1 The first argument now requires a 'plain object' (in jQuery parlance) ie. just {} or Object.create(null)

  • v1.1.0 Added support for standard currying of remaining function arguments

  • v1.0.0 Initial functionality

License - MIT

Copyright (C) 2013 Jon-Carlos Rivera

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.

About

Curry-like secret sauce for option-object accepting function

Resources

License

Stars

Watchers

Forks

Packages

No packages published