Mongoose: 基本モデルは選択を尊重しません:弁別子の場合はfalse

作成日 2017年02月16日  ·  3コメント  ·  ソース: Automattic/mongoose

機能をリクエストしバグを報告しますか?
バグ

現在の動作は何ですか?
ベースモデルを使用してディスクリミネーターモデルドキュメントを取得する場合、 select: falseは尊重されません。

現在の動作がバグである場合は、再現する手順を提供してください。

const mongoose = require('mongoose');
mongoose.Promise = global.Promise;
const co = require('co');
const chalk = require('chalk');

const GITHUB_ISSUE = `test-constructor-emit-bug`;

exec()
  .then(() => {
    console.log(chalk.green(`Successfully ran program`));
    process.exit(0);
  })
  .catch(error => {
    console.log(chalk.red(`Error: ${ error }\n${ error.stack }`));
    process.exit(2);
  });

function exec() {
  return co(function* () {
    const db = mongoose.createConnection(`mongodb://localhost:27017/${ GITHUB_ISSUE }`);
    const { Base, Discriminator } = createModels(db);
    const { baseDoc, discrimDoc } = yield seedDb({ Base, Discriminator });

    const docFromDb = yield Base.findById(discrimDoc._id) // does not honor select: false
    // const docFromDb = yield Discriminator.findById(discrimDoc._id); // honors select: false
    console.log(docFromDb.internal); // should not log `internal.password`
  });
}

function seedDb(models) {
  return co(function*() {
    const { Base, Discriminator } = models;

    yield [Base.remove({}), Discriminator.remove({})];

    const baseDoc = yield Base.create({ internal: { diseases: ['Malaria'] }});
    const discrimDoc = yield Discriminator.create({ internal: {
      diseases: ['MS'],
      password: 'plain_test_password_ftw'
    }});

    return { baseDoc, discrimDoc };
  });
}

function createModels(db) {
  const baseSchema = new mongoose.Schema({
    internal: {
      diseases: [{ type: String }]
    }
  });

  const Base = db.model('Base', baseSchema);
  const discriminatorSchema = new mongoose.Schema({
    internal: {
      password: { type: String, select: false }
    }
  });
  const Discriminator = Base.discriminator('Discriminator', discriminatorSchema);

  return { Base, Discriminator };
}

期待される動作は何ですか?
マングースはどの弁別子を選択するかを事前に知らないため、データベースからフェッチした後、これらのフィールドの選択を手動で解除する必要があります。

node.js、mongoose、MongoDBのバージョンを教えてください。
ノード6.9.2、マングース4.8.3、mongodb 3.4

confirmed-bug

全てのコメント3件

この変更により、予期しない副作用が発生しているようです。

ベースモデルAと、ディスクリミネーターモデルBます。
$slice配列射影を使用してドキュメントをAにクエリすると、タイプA結果のドキュメントはすべてのプロパティ( select: falseとしてマークされたものを除く)を返しますが、タイプB結果のドキュメントはselect: trueとして_明示的に_マークされたプロパティまたはdefault設定されたプロパティのみを返します。 default設定されているものは、データベースに実際に格納されている値ではなく、デフォルト値を返します。 select: trueでマークされているプロパティがない場合は、 default値を持つプロパティのみが返されます(ストア値ではなく、デフォルト値が返されます)。

編集:私のコメントを無視してください、問題はmongooseが内部のクエリに識別子キーを追加することでした。 解決策は、クエリでslice関数を使用すること

話が早すぎたかもしれません。 バージョン4.13.4と最新のmasterコミットの両方でテストすると、 sliceプロジェクションのみを使用した場合に同じ動作が見られます。

私はこの号の元の例を取り上げ、動作を再現するように変更しました。

const mongoose = require('mongoose');
mongoose.Promise = global.Promise;
const co = require('co');
const chalk = require('chalk');
mongoose.set('debug', true);
const GITHUB_ISSUE = `default-values-on-discriminator-documents`;

exec()
    .then(() => {
        console.log(chalk.green(`Successfully ran program`));
        process.exit(0);
    })
    .catch(error => {
        console.log(chalk.red(`Error: ${error}\n${error.stack}`));
        process.exit(2);
    });

function exec() {
    return co(function* () {
        const db = mongoose.createConnection(`mongodb://localhost:27017/${GITHUB_ISSUE}`);
        const { Base, Discriminator } = createModels(db);
        const { baseDoc, discrimDoc } = yield seedDb({ Base, Discriminator });

        const baseDocsFromDbWithSlice = yield Base.find().slice('array', 1)
        const baseDocsFromDb = yield Base.find()
        console.log(baseDocsFromDbWithSlice); // Discriminator document returns default value for propA (and not propB), base document returns stored value - unexpected
        console.log(baseDocsFromDb); // Discriminator and base document returns stored value for propA - expected
    });
}

function seedDb(models) {
    return co(function* () {
        const { Base, Discriminator } = models;

        yield [Base.remove({}), Discriminator.remove({})];

        const baseDoc = yield Base.create({
            propA: 'Hi',
            array: ["a", "b"]
        });
        const discrimDoc = yield Discriminator.create({
            propA: 'Hi',
            propB: 'Hello',
            array: ["a", "b"]
        });

        return { baseDoc, discrimDoc };
    });
}

function createModels(db) {
    const baseSchema = new mongoose.Schema({
        propA: { type: String, default: 'default value' },
        array: [{type: String}],
    });

    const Base = db.model('Base', baseSchema);
    const discriminatorSchema = new mongoose.Schema({
        propB: { type: String}
    });
    const Discriminator = Base.discriminator('Discriminator', discriminatorSchema);

    return { Base, Discriminator };
}

この動作は4.11.13で変更されたため、この問題の解決策が原因であると考えました。 そうでない場合は、新しい問題を作成させていただきます

@bruunを報告していただきありがとうございます、できるだけ早く調査します:+1:

このページは役に立ちましたか?
0 / 5 - 0 評価

関連する問題

adamreisnz picture adamreisnz  ·  3コメント

Igorpollo picture Igorpollo  ·  3コメント

adamreisnz picture adamreisnz  ·  3コメント

lukasz-zak picture lukasz-zak  ·  3コメント

weisjohn picture weisjohn  ·  3コメント