diff --git a/lib/dao.js b/lib/dao.js index 9289d64c8..bf3d932e3 100644 --- a/lib/dao.js +++ b/lib/dao.js @@ -1109,8 +1109,15 @@ DataAccessObject.findOrCreate = function findOrCreate(query, data, options, cb) let obj; if (data) { - obj = new Model(data, {fields: query.fields, applySetters: false, - persisted: true}); + const ctorOpts = { + fields: query.fields, + applySetters: false, + persisted: true, + }; + if (Model.settings.applyDefaultsOnReads === false) { + ctorOpts.applyDefaultValues = false; + } + obj = new Model(data, ctorOpts); } if (created) { @@ -1632,6 +1639,9 @@ DataAccessObject.find = function find(query, options, cb) { applySetters: false, persisted: true, }; + if (Model.settings.applyDefaultsOnReads === false) { + ctorOpts.applyDefaultValues = false; + } let obj; try { obj = new Model(data, ctorOpts); diff --git a/test/basic-querying.test.js b/test/basic-querying.test.js index 2610689cd..1e195d7e3 100644 --- a/test/basic-querying.test.js +++ b/test/basic-querying.test.js @@ -922,6 +922,60 @@ describe('basic-querying', function() { }); }); }); + + it('applies default values by default', async () => { + // Backwards compatibility, see + // https://github.com/strongloop/loopback-datasource-juggler/issues/1692 + + // Initially, all Products were always active, no property was needed + const Product = db.define('Product', {name: String}); + await db.automigrate('Product'); + const created = await Product.create({name: 'Pen'}); + + // Later on, we decide to introduce `active` property + Product.defineProperty('active', { + type: Boolean, + default: false, + }); + + // And query existing data + const found = await Product.findOne(); + found.toObject().should.eql({ + id: created.id, + name: 'Pen', + // Backwards-compatibility + // When Pen does not have "active" flag set, we change it to default + active: false, + }); + }); + + it('preserves empty values from the database when "applyDefaultsOnReads" is false', async () => { + // https://github.com/strongloop/loopback-datasource-juggler/issues/1692 + + // Initially, all Products were always active, no property was needed + const Product = db.define( + 'Product', + {name: String}, + {applyDefaultsOnReads: false}, + ); + + await db.automigrate('Product'); + const created = await Product.create({name: 'Pen'}); + + // Later on, we decide to introduce `active` property + Product.defineProperty('active', { + type: Boolean, + default: false, + }); + + // And query existing data + const found = await Product.findOne(); + found.toObject().should.eql({ + id: created.id, + name: 'Pen', + active: undefined, + }); + }); }); describe('count', function() { diff --git a/test/manipulation.test.js b/test/manipulation.test.js index 9c939c0d6..5d355d4b5 100644 --- a/test/manipulation.test.js +++ b/test/manipulation.test.js @@ -1619,6 +1619,60 @@ describe('manipulation', function() { }) .catch(done); }); + + it('applies default values on returned data', async () => { + // Backwards compatibility, see + // https://github.com/strongloop/loopback-datasource-juggler/issues/1692 + + // Initially, all Products were always active, no property was needed + const Product = db.define('Product', {name: String}); + await db.automigrate('Product'); + const created = await Product.create({name: 'Pen'}); + + // Later on, we decide to introduce `active` property + Product.defineProperty('active', { + type: Boolean, + default: false, + }); + + // and findOrCreate an existing record + const [found] = await Product.findOrCreate({id: created.id}, {name: 'updated'}); + found.toObject().should.eql({ + id: created.id, + name: 'Pen', + // Backwards-compatibility + // When Pen does not have "active" flag set, we change it to default + active: false, + }); + }); + + it('preserves empty values from the database when "applyDefaultsOnReads" is false', async () => { + // https://github.com/strongloop/loopback-datasource-juggler/issues/1692 + + // Initially, all Products were always active, no property was needed + const Product = db.define( + 'Product', + {name: String}, + {applyDefaultsOnReads: false}, + ); + + await db.automigrate('Product'); + const created = await Product.create({name: 'Pen'}); + + // Later on, we decide to introduce `active` property + Product.defineProperty('active', { + type: Boolean, + default: false, + }); + + // And findOrCreate an existing record + const [found] = await Product.findOrCreate({id: created.id}, {name: 'updated'}); + found.toObject().should.eql({ + id: created.id, + name: 'Pen', + active: undefined, + }); + }); }); describe('destroy', function() { diff --git a/types/model.d.ts b/types/model.d.ts index f16da3fe4..4e31ce88c 100644 --- a/types/model.d.ts +++ b/types/model.d.ts @@ -76,6 +76,13 @@ export interface ModelProperties { export interface ModelSettings extends AnyObject { strict?: boolean; forceId?: boolean; + + /* + * Apply default property values to data fetched from database. + * Enabled by default. + * TODO(semver-major): disable by default + */ + applyDefaultsOnReads?: boolean; } /**