Minor changes before v1.0.0 can include breaking changes.
- Fixed map selectors being stored under the same key. (ad9d0c7)
- Prevent map selectors from returning unrelated refs. (27bddb6)
- Removed unnecessary branches (#316)
- Optimized
objectShallowEquals
usingevery
instead offorEach
(#317)
This release should not be breaking. It is a new minor version only to safeguard from potential edge cases.
Fix memoization for map selectors. (#318)
Consider the selector createSelector(orm.Book.authors.map(orm.Author.name))
. When the list of a book's authors didn't change but one of the authors' names changed, the selector would still return the old name.
- Serializing the database using
JSON.stringify
is now officially supported and tested. This allows using redux-persist without any special configuration again. We no longer use a symbol to detect state objects, so you may need to add a migration to make this work with older persisted states. (#321) - Fixed
'desc'
values for ordering query sets. (#311) - Fixed a bug with field selectors for self-referencing fields not being able to infer the model that the fields point to. (#315)
- Use
Symbol.for
to create symbols in the global symbol registry. (#290)
Allows you to use Redux-ORM across JavaScript contexts again.
- Much easier selectors for common use-cases. (#257)
- Specify related models using Model classes instead of strings. (#266)
- Pass custom accessor names for many-to-many fields to be able to access initially passed data. (#279)
- Prevent caching selectors if individual records accessed via index have changed. (#255, 02edce9)
- Also filter by non-PK attributes when user specifies a PK in filter object. (69a87f8)
- Replace usage of
lodash/mapValues
by ES8Object.entries.reduce
.
Optimized full bundles using babel-plugin-lodash
and lodash-webpack-plugin
:
dist/redux-orm.min.js
decreased from 67.7 kB to 48.5 kB.- Gzipped size decreased from 18.5 kB to 13.1 kB.
- Don't filter needlessly after applying foreign key indexes. Prevents applying object-based filters again after already fully satisfying their conditions using indexes. (41ece84)
- Remove attributes already looked up via index from filter clauses. After applying indexes, if additional filtering needs to occur, strip away the indexed columns from the filter object. (278ff6f)
Removed tests, coverage reports and other unneeded files from NPM package builds. This decreased our NPM package size to around 700 kB.
- Automatically index foreign keys. Queries will be fulfilled by using foreign key indexes if possible. Only works for direct FK relationships. For instance,
book.authors
will be optimized if theAuthor
model has afk({ to: 'Book', 'relatedName': 'authors' })
field. (#250, b9c1635)
This release should be non-breaking, however it does slightly decrease performance of writes. Many-to-many queries (e.g. child.parents
) are also less performant, but in theory it is now possible to optimize them by an order of magnitude (similar to SQL joins).
- Fixed a bug with memoization when specifying a custom
mapName
usingModel.options
. (0fb2aca) - Optimized memoization by not scanning entire tables for equality but relying on the propagated table identity instead. (0fb2aca)
- Optimized model initialization by using
Object.keys()
instead ofObject.entries()
. (e371ac7)
- Updated dependencies, most notably Babel v7.
- Set
sideEffects: false
inpackage.json
for tree-shaking. - Replaced some internal usage of lodash by ES6 methods.
- Fixed self-referencing many-to-many relationships. They previously caused errors during initialization. (#226)
- Throw error when user attempts to register a model class without a
modelName
set. (#231) - Throw error when user attempts to interact with the database without a
session
. (#237) - Fixed tests not running on Windows. (abe8723)
- Drastically increased test coverage and split up tests to make them easier to digest. (#220, 294d5f2)
- Removed unused methods from Model. (f9efb8b)
- Our exported ES module in
es/
(pkg.module
) is now ES5 compatible. (#216, #221) - You can supply an
as
option to foreign key fields (fk()
) to have Redux-ORM create a separate field for their accessors so as to not override foreign key references (#219):
class Movie extends Model {};
Movie.fields = {
publisherId: fk({
to: 'Publisher',
as: 'publisher',
relatedName: 'movies',
}),
};
Publisher.create({ id: 123 });
const movie = Movie.create({ publisherId: 123 });
movie.publisherId // 123
movie.publisher.id === movie.publisherId // true
- Fixed bug where using the ES5 version with ES6-based model classes would not work. #213
- Unfortunately this hotfix contains a performance degradation, so only upgrade if you suffer from the above issue. We are working on an alternative solution.
-
Potential breaking changes to model API:
Model#equals
is now used byModel#update
to prevent unnecessary updates in some cases when the intended updates would not change the current model. By default models will always be updated if any of the passed attributes have a different reference than the respective existing attribute. #204Model.withId
and resolving foreign key relationships no longer throw an error if the referenced model does not exist. Insteadnull
is returned. #210Model.exists
now accepts passing a lookup object to check if any model instance matching the passed attributes exists. #209Model.hasId
has been replaced byModel.idExists
. #209
-
Fixed
QuerySet#toString
. 12ce882 -
Upgraded to reselect v3.0.1. #208
-
Refactored
fields.js
,descriptors.js
and small parts ofModel.js
. #210
pkg.module
now points to a new entrypoint ates/index.js
adhering to the latest finalized ECMAScript version without non-standard language features. 3ab92b9 To use the original source in your application, please have your bundler resolvesrc/index.js
as defined inpkg["jsnext:main"]
.
- Fixed filter functions not working when using strings as id attributes. #207
- Added memoization by instance. Prevents unnecessary full table scans when not interested in unchanged records. #185
- Improved performance of many-to-many descriptors. #165
- Faster model lookups by primary key (id). #158
- Added
@deprecated
JSDoc tag to deprecated members and functions. #201 - Added
pkg.module
path topackage.json
indicating path for ES6 module. May fix #53 for some people. #192 - Introduced dependency on ES2015
Set
. See Polyfill. - Added performance regression tests.
fix for filter/query #99
fix for many-many updates #136
- Added 'upsert' method to Model (insert or update behaviour)
- Major updates for tests
- CI integration
A lot. See the migration guide.
Adds UMD build to partially fix #41. You can now use or try out redux-orm
through a script tag:
<script src="https://unpkg.com/redux-orm/dist/redux-orm.min.js"></script>
redux-orm.js
will point to the master version of the library; If you need to stick to a version, make a copy or build it yourself.
Fixed bug that mutated the backend options in Model
if you supplied custom ones, see Issue 37. Thanks to @diffcunha for the fix!
Fixed regression in Model.prototype.update
Added babel-runtime to dependencies
Adds batched mutations. This is a big performance improvement. Previously adding 10,000 objects would take 15s, now it takes about 0.5s. Batched mutations are implemented using immutable-ops
internally.
Breaking changes:
- Removed
indexById
option from Backend. This means that data will always be stored in both an array of id's and a map ofid => entity
, which was the default setting. If you didn't explicitly setindexById
tofalse
, you don't need to change anything. - Batched mutations brought some internal changes. If you had custom
Backend
orSession
classes, or have overriddenModel.getNextState
, please check out the diff.
Breaking changes:
Model classes that you access in reducers and selectors are now session-specific. Previously the user-defined Model class reference was used for sessions, with a private session
property changing based on the most recently created session. Now Model classes are given a unique dummy subclass for each session. The subclass will be bound to that specific session. This allows multiple sessions to be used at the same time.
You most likely don't need to change anything. The documentation was written with this feature in mind from the start. As long as you've used the model class references given to you in reducers and selectors as arguments (not the reference to the model class you defined), you're fine.
Breaking changes:
- When calling
QuerySet.filter
orQuerySet.exclude
with an object argument, any values of that object that look like aModel
instance (i.e. they have agetId
property that is a function), will be turned into the id of that instance before performing the filtering or excluding.
E.g.
Book.filter({ author: Author.withId(0) });
Is equivalent to
Book.filter({ author: 0 });
Breaking changes:
- Model instance method
equals(otherModel)
now checks if the two model's attributes are shallow equal. Previously, it checked if the id's and model classes are equal. - Session constructor now receives a Schema instance as its first argument, instead of an array of Model classes (this only affects you if you're manually instantiating Sessions with the
new
operator).
Other changes:
- Added
hasId
static method to the Model class. It tests for the existence of the supplied id in the model's state. - Added instance method
getNextState
to the Session class. This enables you to get the next state without running model-reducers. Useful if you're bootstrapping data, writing tests, or otherwise operating on the data outside reducers. You can pass an options object that currently accepts arunReducers
key. It's value indicates if reducers should be run or not. - Improved API documentation.
- Fixed a bug that mutated props passed to Model constructors, which could be a reference from the state. I highly recommend updating from 0.3.1.
- API cleanup, see breaking changes below.
- Calling getNextState is no longer mandatory in your Model reducers. If your reducer returns
undefined
,getNextState
will be called for you.
Breaking changes:
- Removed static methods
Model.setOrder()
andBackend.order
. If you want ordered entities, use the QuerySet instance methodorderBy
. - Added helpful error messages when trying to add a duplicate many-to-many entry (Model.someManyRelated.add(...)), trying to remove an unexisting many-to-many entry (Model.exampleManyRelated.remove(...)), or creating a Model with duplicate many-to-many entry ids (Model.create(...)).
- Removed ability to supply a mapping function to QuerySet instance method
update
. If you need to record updates dynamically based on each entity, iterate through the objects withforEach
and record updates separately:
const authors = publisher.authors;
authors.forEach(author => {
const isAdult = author.age >= 18;
author.update({ isAdult });
})
or use the ability to merge an object with all objects in a QuerySet. Since the update operation is batched for all objects in the QuerySet, it can be more performant with a large amount of entities:
const authors = publisher.authors;
const isAdult = author => author.age >= 18;
const adultAuthors = authors.filter(isAdult);
adultAuthors.update({ isAdult: true });
const youngAuthors = authors.exclude(isAdult);
youngAuthors.update({ isAdult: false });
A descriptive error is now thrown when a reverse field conflicts with another field declaration. For example, the following schema:
class A extends Model {}
A.modelName = 'A';
class B extends Model {}
B.modelName = 'B';
B.fields = {
field1: one('A'),
field2: one('A'),
};
would try to define the reverse field b
on A
twice, throwing an error with an undescriptive message.
Breaking changes:
Model.withId(id)
now throws if object with idid
does not exist in the database.
Includes various bugfixes and improvements.
Breaking changes:
- Replaced
plain
andmodels
instance attributes inQuerySet
withwithRefs
andwithModels
respectively. The attributes return a newQuerySet
instead of modifying the existing one. Aref
alias is also added forwithRefs
, so you can doBook.ref.at(2)
. - After calling
filter
,exclude
ororderBy
method on aQuerySet
instance, thewithRefs
flag is always flipped off so that calling the same methods on the returnedQuerySet
would use model instances in the operations. Previously the flag value remained after calling those methods. .toPlain()
fromQuerySet
is renamed to.toRefArray()
for clarity.- Added
.toModelArray()
method toQuerySet
. - Removed
.objects()
method fromQuerySet
. Use.toRefArray()
or.toModelArray()
instead. - Removed
.toPlain()
method fromModel
, which returned a copy of the Model instance's property values. To replace that,ref
instance getter was added. It returns a reference to the plain JavaScript object in the database. So you can doBook.withId(0).ref
. If you need a copy, you can doObject.assign({}, Book.withId(0).ref)
. - Removed
.fromEmpty()
instance method fromSchema
. - Removed
.setReducer()
instance method fromSchema
. You can just doModelClass.reducer = reducerFunc;
.