Используйте включенные модели в выражениях Where с Sequelize - PullRequest
0 голосов
/ 12 сентября 2018

Я использую Sequelize 4.38.0 с Postgres.Я пытаюсь реализовать функцию поиска.Но у меня проблемы с проверкой запроса.Я должен проверить, соответствует ли запрос городу.Это не обязательно должно совпадать.

Я хочу переместить запрос к модели города на верхний уровень ГДЕ, как описано здесь: http://docs.sequelizejs.com/manual/tutorial/models-usage.html#top-level-where-with-eagerly-loaded-models Но я получаю эту ошибку: Unhandled rejection SequelizeDatabaseError: missing an entry for the "City" table in the FROM clause

Определение модели

const Property = sequelize.define('property', {
  id: {
    type: type.STRING,
    unique: true,
    allowNull: false,
    primaryKey: true
  },
  name: {
    type: type.STRING,
    allowNull: false
  },
  // ...
  isVisible: {
    type: type.BOOLEAN,
    default: false
  }
});


const City = sequelize.define('city', {
  name: {
    type: type.STRING
  },
}, {
  timestamps: false,
});

Запрос

sequelize.sync().then(() => {
  let query = 'new';
  let whereStatement = {
    $or: [{
        name: {
          $ilike: '%' + query + '%'
        }
      },
      {
        description: {
          $ilike: '%' + query + '%'
        }
      },
      // I get the Error if I uncomment this
      // { 
      //   '$City.name$': {
      //     $ilike: '%' + query + '%'
      //   }
      // }
    ]
  };

  Property.findAndCountAll({
    where: whereStatement,
    include: [{
        model: City,
        where: {
          name: {
            $ilike: '%' + query + '%'
          }
        }
        // I want to move this where to the upper level
        // where inside the OR opertator
      },
      {
        model: Photo
      }
    ],
    distinct: true,
    order: [
      ['createdAt', 'DESC']
    ]
  }).then(properties => {
    console.log(properties.count)
  });
});

Фактический запрос имеет немного больше атрибутов.Результирующий SQL:

Executing (default): SELECT "property".*, "city"."id" AS "city.id", "city"."name" AS "city.name", "city"."provinceId" AS "city.provinceId", "photos"."id" AS "photos.id", "photos"."location" AS "photos.location", "photos"."slug" AS "photos.slug", "photos"."description" AS "photos.description", "photos"."isPrimary" AS "photos.isPrimary", "photos"."createdAt" AS "photos.createdAt", "photos"."updatedAt" AS "photos.updatedAt", "photos"."propertyId" AS "photos.propertyId" FROM (SELECT "property"."id", "property"."webSlug", "property"."name", "property"."description", "property"."direction", "property"."price", "property"."areaConstruction", "property"."areaLot", "property"."isVisible", "property"."isAvailable", "property"."isNegociable", "property"."isRural", "property"."latitude", "property"."longitude", "property"."map_radio", "property"."video_url", "property"."createdAt", "property"."updatedAt", "property"."cityId", "property"."propertyTypeId" FROM "properties" AS "property" WHERE ("property"."name" ILIKE '%My Query%' OR "property"."description" ILIKE '%My Query%' OR "property"."id" ILIKE '%My Query%' OR "properties_city"."name" ILIKE '%My Query%') AND "property"."isVisible" = true ORDER BY "property"."createdAt" DESC LIMIT 10 OFFSET 0) AS "property" LEFT OUTER JOIN "cities" AS "city" ON "property"."cityId" = "city"."id" LEFT OUTER JOIN "photos" AS "photos" ON "property"."id" = "photos"."propertyId" ORDER BY "property"."createdAt" DESC;

Обновление

Если я использую те же Модели и те же whereStatement, но я изменяю функцию findAndCountAll() на findAll() поискЭта функция работает правильно, и я могу добавить '$city.name$': { $ilike: '%' + query + '%' } в верхний WHERE.

Property.findAll({
    where: whereStatement,
    include: [{model: City}, {model: Photo}],
    distinct: true,
    order: [['createdAt', 'DESC']],
  }).then(properties => {
    res.render('search', {
      title: 'Buscar',
      properties: properties,
      query: query,
      propertyTypeId: propertyTypeId,
      propertyTypes: req.propertyTypes,
    })
  })

Вывод SQL:

Executing (default): SELECT "property"."id", "property"."webSlug", "property"."name", "property"."description", "property"."direction", "property"."price", "property"."areaConstruction", "property"."areaLot", "property"."isVisible", "property"."isAvailable", "property"."isNegociable", "property"."isRural", "property"."latitude", "property"."longitude", "property"."map_radio", "property"."video_url", "property"."createdAt", "property"."updatedAt", "property"."cityId", "property"."propertyTypeId", "city"."id" AS "city.id", "city"."name" AS "city.name", "city"."provinceId" AS "city.provinceId", "photos"."id" AS "photos.id", "photos"."location" AS "photos.location", "photos"."slug" AS "photos.slug", "photos"."description" AS "photos.description", "photos"."isPrimary" AS "photos.isPrimary", "photos"."createdAt" AS "photos.createdAt", "photos"."updatedAt" AS "photos.updatedAt", "photos"."propertyId" AS "photos.propertyId" FROM "properties" AS "property" LEFT OUTER JOIN "cities" AS "city" ON "property"."cityId" = "city"."id" LEFT OUTER JOIN "photos" AS "photos" ON "property"."id" = "photos"."propertyId" WHERE ("property"."name" ILIKE '%City_Query%' OR "property"."description" ILIKE '%City_Query%' OR "property"."id" ILIKE '%City_Query%' OR "city"."name" ILIKE '%City_Query%') AND "property"."isVisible" = true ORDER BY "property"."createdAt" DESC;

1 Ответ

0 голосов
/ 13 сентября 2018

Попробуйте, это должно работать.включить

"дублирование": ложь

в этом

sequelize.sync().then(() => {
      let query = 'new';
      let whereStatement = {
        $or: [{
            name: {
              $ilike: '%' + query + '%'
            }
          },
          {
            description: {
              $ilike: '%' + query + '%'
            }
          },
          { 
             '$City.name$': {
               $ilike: '%' + query + '%'
             }
          }
        ]
      };

      Property.findAndCountAll({
        where: whereStatement,
        include: [{
            model: City,
            attributes: [],
            duplicating:false
          },
          {
            model: Photo
          }
        ],
        distinct: true,
        order: [
          ['createdAt', 'DESC']
        ]
      }).then(properties => {
        console.log(properties.count)
      });
    });
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...