From b6a452b95f5065d096b76c3ac657948ea321c3a0 Mon Sep 17 00:00:00 2001 From: sdjdd Date: Tue, 2 Apr 2024 19:10:44 +0800 Subject: [PATCH] fix(next/api): ES multi_match query cannot match nested field --- next/api/src/service/search-ticket.ts | 27 +++++++++++++++++++-------- 1 file changed, 19 insertions(+), 8 deletions(-) diff --git a/next/api/src/service/search-ticket.ts b/next/api/src/service/search-ticket.ts index 0128bd435..c3934f6f1 100644 --- a/next/api/src/service/search-ticket.ts +++ b/next/api/src/service/search-ticket.ts @@ -1,6 +1,6 @@ import { Client } from '@elastic/elasticsearch'; import { Queue } from 'bull'; -import esb from 'elastic-builder'; +import esb, { Query as ESQuery } from 'elastic-builder'; import _ from 'lodash'; import { @@ -285,15 +285,23 @@ export class SearchTicketService { filters.fields.forEach((match) => addNestedQuery('fields', match)); } if (filters.keyword) { - const terms = this.parseTerms(filters.keyword); - terms.forEach((term) => { - const matchQuery = esb - .multiMatchQuery(['title', 'content', 'fields.value'], term.value) - .operator('and'); + this.parseTerms(filters.keyword).forEach((term) => { + let normalMatchQuery: ESQuery; + let nestedMatchQuery: ESQuery; if (term.isPhrase) { - matchQuery.type('phrase'); + normalMatchQuery = esb.multiMatchQuery(['title', 'content'], term.value).type('phrase'); + nestedMatchQuery = esb.nestedQuery( + esb.matchPhraseQuery('fields.value', term.value), + 'fields' + ); + } else { + normalMatchQuery = esb.multiMatchQuery(['title', 'content'], term.value).operator('and'); + nestedMatchQuery = esb.nestedQuery( + esb.matchQuery('fields.value', term.value).operator('and'), + 'fields' + ); } - boolQuery.filter(matchQuery); + boolQuery.filter(esb.boolQuery().should([normalMatchQuery, nestedMatchQuery])); }); } @@ -307,6 +315,9 @@ export class SearchTicketService { .source(false) .toJSON(); + console.log('[Search Ticket]: Query:'); + console.dir(body, { depth: 10 }); + const res = await this.esClient.search({ index: this.indexName, body,