Security News
UK Officials Consider Banning Ransomware Payments from Public Entities
The UK is proposing a bold ban on ransomware payments by public entities to disrupt cybercrime, protect critical services, and lead global cybersecurity efforts.
seamless-immutable
Advanced tools
Immutable data structures for JavaScript which are backwards-compatible with normal JS Arrays and Objects.
The seamless-immutable npm package provides a way to create and work with deeply immutable data structures in JavaScript. It ensures that objects and arrays cannot be modified after they are created, which helps in maintaining predictable state and avoiding side effects in applications.
Creating Immutable Objects
This feature allows you to create immutable objects. Once created, the object cannot be modified.
const Immutable = require('seamless-immutable');
const obj = Immutable({ name: 'John', age: 30 });
console.log(obj); // Output: { name: 'John', age: 30 }
Creating Immutable Arrays
This feature allows you to create immutable arrays. Once created, the array cannot be modified.
const Immutable = require('seamless-immutable');
const arr = Immutable([1, 2, 3]);
console.log(arr); // Output: [1, 2, 3]
Merging Immutable Objects
This feature allows you to merge immutable objects, creating a new immutable object with the merged properties.
const Immutable = require('seamless-immutable');
const obj1 = Immutable({ name: 'John', age: 30 });
const obj2 = obj1.merge({ age: 31 });
console.log(obj2); // Output: { name: 'John', age: 31 }
Updating Immutable Arrays
This feature allows you to perform operations on immutable arrays, resulting in a new immutable array.
const Immutable = require('seamless-immutable');
const arr = Immutable([1, 2, 3]);
const newArr = arr.map(x => x * 2);
console.log(newArr); // Output: [2, 4, 6]
The 'immutable' package provides persistent immutable data structures including List, Stack, Map, OrderedMap, Set, and OrderedSet. It offers more complex data structures and methods compared to seamless-immutable, but can be more complex to use.
The 'immer' package allows you to work with immutable state by using a 'draft' state that you can modify directly. It then produces the next immutable state based on the changes. It is simpler to use for complex state updates compared to seamless-immutable.
The 'mori' package provides a set of Clojure-inspired immutable data structures and functions for JavaScript. It is more functional programming-oriented and offers a different approach compared to seamless-immutable.
Immutable JS data structures which are backwards-compatible with normal Arrays and Objects.
Use them in for
loops, pass them to functions expecting vanilla JavaScript data structures, etc.
var array = Immutable(["totally", "immutable", {hammer: "Can’t Touch This"}]);
array[1] = "I'm going to mutate you!"
array[1] // "immutable"
array[2].hammer = "hm, surely I can mutate this nested object..."
array[2].hammer // "Can’t Touch This"
for (var index in array) { console.log(array[index]); }
// "totally"
// "immutable"
// { hammer: 'Can’t Touch This' }
JSON.stringify(array) // '["totally","immutable",{"hammer":"Can’t Touch This"}]'
This level of backwards compatibility requires ECMAScript 5 features like Object.defineProperty and Object.freeze to exist and work correctly, which limits the browsers that can use this library to the ones shown in the test results below. (tl;dr IE9+)
Whenever you deeply clone large nested objects, it should typically go much faster with Immutable
data structures. This is because the library reuses the existing nested objects rather than instantiating new ones.
In the development build, objects are frozen. (Note that Safari is relatively slow to iterate over frozen objects.) The development build also overrides unsupported methods (methods that ordinarily mutate the underlying data structure) to throw helpful exceptions.
The production (minified) build does neither of these, which significantly improves performance.
Immutable()
returns a backwards-compatible immutable representation of whatever you pass it, so feel free to pass it absolutely anything that can be serialized as JSON. (As is the case with JSON, objects containing circular references are not allowed. Functions are allowed, unlike in JSON, but they will not be touched.)
Since numbers, strings, undefined
, and null
are all immutable to begin with, the only unusual things it returns are Immutable Arrays and Immutable Objects. These have the same ES5 methods you’re used to seeing on them, but with these important differences:
ImmutableError
.foo[5] = bar
) will not work. Browsers other than Internet Explorer will throw a TypeError
if use strict is enabled, and in all other cases it will fail silently.For example:
Immutable([3, 1, 4]).sort()
// This will throw an ImmutableError, because sort() is a mutating method.
Immutable([1, 2, 3]).concat([10, 9, 8]).sort()
// This will also throw ImmutableError, because an Immutable Array's methods
// (including concat()) are guaranteed to return other immutable values.
[1, 2, 3].concat(Immutable([6, 5, 4])).sort()
// This will succeed, and will yield a sorted mutable array containing
// [1, 2, 3, 4, 5, 6], because a vanilla array's concat() method has
// no knowledge of Immutable.
Immutable({all: "your base", are: {belong: "to them"}}).merge({are: {belong: "to us"}})
// This handy new method will return the following:
// Immutable({all: "your base", are: {belong: "to us"}})
Like a regular Array, but immutable! You can construct these by passing
an array to Immutable()
:
Immutable([1, 2, 3])
// An immutable array containing 1, 2, and 3.
Beyond the usual Array fare, the following methods have been added.
Immutable(["here", "we", "go"]).flatMap(function(str) {
return [str, str, str];
});
// returns Immutable(["here", "here", "here", "we", "we", "we", "go", "go", "go"])
Immutable(["drop the numbers!", 3, 2, 1, 0, null, undefined]).flatMap(function(value) {
if (typeof value === "number") {
return [];
} else {
return value;
}
});
// returns Immutable(["drop the numbers!", null, undefined])
Effectively performs a map over the elements in the array, except that whenever the provided iterator function returns an Array, that Array's elements are each added to the final result.
Immutable(["hey", "you"]).asObject(function(str) {
return [str, str.toUpperCase()];
});
// returns Immutable({hey: "HEY", you: "YOU"})
Effectively performs a map over the elements in the array, expecting that the iterator function will return an array of two elements - the first representing a key, the other a value. Then returns an Immutable Object constructed of those keys and values.
You can also call .asObject
without passing an iterator, in which case it will proceed assuming the Array
is already organized as desired.
var mutableArray = Immutable(["hello", "world"]).asMutable();
mutableArray.push("!!!");
mutableArray // ["hello", "world", "!!!"]
Returns a mutable copy of the array. For a deeply mutable copy, in which any instances of Immutable
contained in nested data structures within the array have been converted back to mutable data structures, call .asMutable({deep: true})
instead.
Like a regular Object, but immutable! You can construct these by passing an
object to Immutable()
.
Immutable({foo: "bar"})
// An immutable object containing the key "foo" and the value "bar".
To construct an Immutable Object with a custom prototype, simply specify the
prototype in options
(while useful for preserving prototypes, please note
that custom mutator methods will not work as the object will be immutable):
function Square(length) { this.length = length };
Square.prototype.area = function() { return Math.pow(this.length, 2) };
Immutable(new Square(2), {prototype: Square.prototype}).area();
// An immutable object, with prototype Square,
// containing the key "length" and method `area()` returning 4
Beyond the usual Object fare, the following methods have been added.
Immutable({status: "good", hypothesis: "plausible", errors: 0}).merge({status: "funky", hypothesis: "confirmed"})
// returns Immutable({status: "funky", hypothesis: "confirmed", errors: 0})
Immutable({status: "bad", errors: 37}).merge([
{status: "funky", errors: 1}, {status: "groovy", errors: 2}, {status: "sweet"}])
// returns Immutable({status: "sweet", errors: 2})
// because passing an Array (or just multiple arguments) is shorthand for
// invoking a separate merge for each object in turn.
Returns an Immutable Object containing the properties and values of both this object and the provided object, prioritizing the provided object's values whenever the same key is present in both objects.
Multiple objects can be provided in an Array in which case more merge
invocations will be performed using each provided object in turn.
A second argument can be provided to perform a deep merge: {deep: true}
.
Immutable({the: "forests", will: "echo", with: "laughter"}).without("with")
// returns Immutable({the: "forests", will: "echo"})
Immutable({the: "forests", will: "echo", with: "laughter"}).without(["will", "with"])
// returns Immutable({the: "forests"})
Immutable({the: "forests", will: "echo", with: "laughter"}).without("will", "with")
// returns Immutable({the: "forests"})
Returns an Immutable Object excluding the given keys from the existing object.
Multiple keys can be provided, either in an Array or as extra arguments.
var mutableObject = Immutable({when: "the", levee: "breaks"}).asMutable();
mutableObject.have = "no place to go";
mutableObject // {when: "the", levee: "breaks", have: "no place to go"}
Returns a mutable copy of the object. For a deeply mutable copy, in which any instances of Immutable
contained in nested data structures within the object have been converted back to mutable data structures, call .asMutable({deep: true})
instead.
Now when you require("seamless-immutable")
, you get the development build by default.
main
now points to src/seamless-immutable.js
so you can more easily build with envify
yourself.
Add support for optional prototyping.
Calling .asMutable({deep: true}) on an Immutable data structure with a nested Date no longer throws an exception.
Arrays with nonstandard prototypes no longer throw exceptions when passed to Immutable
.
Custom mergers now check for reference equality and abort early if there is no more work needed, allowing improved performance.
Fixes a bug where indices passed into iterators for flatMap and asObject were strings instead of numbers.
Fixes an IE and Firefox bug related to cloning Dates while preserving their prototypes.
Dates now retain their prototypes, the same way Arrays do.
Adds a minified production build with no freezing or defensive unsupported methods, for a ~2x performance boost.
Adds optional merger
function to #merge
.
Bugfix: #merge
with {deep: true}
no longer attempts (unsuccessfully) to deeply merge arrays as though they were regular objects.
Minor documentation typo fix.
Breaking API change: #merge
now takes exactly one or exactly two arguments. The second is optional and allows specifying deep: true
.
Don't bother returning a new value from #merge
if no changes would result.
Make error message for invalid #asObject
less fancy, resulting in a performance improvement.
Adds #asMutable
Initial stable release
FAQs
Immutable data structures for JavaScript which are backwards-compatible with normal JS Arrays and Objects.
The npm package seamless-immutable receives a total of 271,957 weekly downloads. As such, seamless-immutable popularity was classified as popular.
We found that seamless-immutable demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 2 open source maintainers 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
The UK is proposing a bold ban on ransomware payments by public entities to disrupt cybercrime, protect critical services, and lead global cybersecurity efforts.
Security News
Snyk's use of malicious npm packages for research raises ethical concerns, highlighting risks in public deployment, data exfiltration, and unauthorized testing.
Research
Security News
Socket researchers found several malicious npm packages typosquatting Chalk and Chokidar, targeting Node.js developers with kill switches and data theft.