Work with text analyzers
The CREATE SEARCH INDEX
DDL
statement,
SEARCH
function, and
TEXT_ANALYZE
function
support advanced text analyzer configuration options. Understanding
BigQuery's text analyzers and their options lets you refine your search
experience.
This document provides an overview of the different text analyzers available in BigQuery and their configuration options, as well as examples of how text analyzers work with search in BigQuery. For more information about text analyzer syntax, see Text analysis.
Text analyzers
BigQuery supports the following text analyzers:
NO_OP_ANALYZER
LOG_ANALYZER
PATTERN_ANALYZER
NO_OP_ANALYZER
Use the NO_OP_ANALYZER
when you have pre-processed data that you want to match
exactly. There is no tokenization or normalization applied to the text. Since
this analyzer does not perform tokenization or normalization, it accepts no
configuration. For more information about
NO_OP_ANALYZER
, see
NO_OP_ANALYZER
.
LOG_ANALYZER
The LOG_ANALYZER
modifies data in the following ways:
- Text is made lowercase.
ASCII values greater than 127 are kept as is.
Text is split into individual terms called tokens by the following delimiters:
[ ] < > ( ) { } | ! ; , ' " * & ? / : = @ . - $ % \ _ \n \r \s \t ! & %26 ; ; | | + + = = %20 ] ] [ [ : : , , ( )
If you don't want to use the default delimiters, you can specify the delimiters you want to use as text analyzer options.
LOG_ANALYZER
lets you configure specific delimiters and token filters for more control over your search results. For more information about the specific configuration options available when using theLOG_ANALYZER
, seedelimiters
analyzer option andtoken_filters
analyzer option.
PATTERN_ANALYZER
The PATTERN_ANALYZER
text analyzer extracts tokens from text using a regular
expression. The regular expression engine and syntax used with
PATTERN_ANALYZER
is RE2. PATTERN_ANALYZER
tokenizes patterns in the following order:
- It finds the first substring that matches the pattern (from the left) in the string. This is a token to be included in the output.
- It removes everything from the input string until the end of the substring found in step 1.
- It repeats the process until the string is empty.
The following table provides examples of PATTERN_ANALYZER
token extraction:
Pattern | Input text | Output tokens |
---|---|---|
ab | ababab |
|
ab | abacad |
|
[a-z]{2} | abacad |
|
aaa | aaaaa |
|
[a-z]/ | a/b/c/d/e |
|
/[^/] / | aa/bb/cc |
|
[0-9] | abc | |
(?:/?)[a-z] | /abc |
|
(?:/)[a-z] | /abc |
|
(?:[0-9]abc){3}(?:[a-z]000){2} | 7abc7abc7abcx000y000 |
|
". " | "cats" and "dogs" |
Note the use of greedy quantifiers makes the match to match the longest string possible in the text, causing '"cats" and "dogs"' to be extracted as a token in the text. |
". ?" | "cats" and "dogs" |
Note the use of lazy quantifiers ? makes the regular expression match the shortest string possible in the text, causing '"cats"', '"dogs"' to be extracted as 2 separate tokens in the text. |
Using the PATTERN_ANALYZER
text analyzer gives you more control over the
tokens extracted from a text when used with the SEARCH
function. The following
table shows how different patterns and results result in different SEARCH
results:
Pattern | Query | Text | Tokens from text | SEARCH(text, query) | Explanation |
---|---|---|---|---|---|
abc | abcdef | abcghi |
|
TRUE | 'abc' in ['abcghi'] |
cd[a-z] | abcdef | abcghi |
|
FALSE | 'cde' in ['abcghi'] |
[a-z]/ | a/b/ | a/b/c/d/ |
|
TRUE | 'a/' in ['a/', 'b/', 'c/', 'd/'] AND 'b/' in ['a/', 'b/', 'c/', 'd/'] |
/[^/] / | aa/bb/ | aa/bb/cc/ |
|
TRUE | '/bb/' in ['/bb/'] |
/[^/] / | bb | aa/bb/cc/ |
|
ERROR | No match found in query term |
[0-9] | abc | abc123 | ERROR | No match found in query term | |
[0-9] | `abc` | abc123 | ERROR | No match found in query term Matching backtick as backtick, not a special character. |
|
[a-z][a-z0-9]*@google\.com | This is my email: [email protected] | [email protected] | TRUE | '[email protected]' in '[email protected]' | |
abc | abc\ abc | abc |
|
TRUE | 'abc' in ['abc'] Note that 'abc abc' is a single subquery(ie) after being parsed by the search query parser since the space is escaped. |
(?i)(?:Abc) (no normalization) | aBcd | Abc |
|
FALSE | 'aBc' in ['Abc'] |
(?i)(?:Abc) normalization: lower_case = true |
aBcd | Abc |
|
TRUE | 'abc' in ['abc'] |
(?:/?)abc | bc/abc | /abc/abc/ |
|
TRUE | '/abc' in ['/abc'] |
(?:/?)abc | abc | d/abc |
|
FALSE | 'abc' in ['/abc'] |
". " | "cats" | "cats" and "dogs" |
|
FALSE | '"cats"' in ['"cats" and "dogs"] Note the use of greedy quantifiers makes the regular expression match the longest string possible in the text, causing '"cats" and "dogs"' to be extracted as a token in the text. |
". ?" | "cats" | "cats" and "dogs" |
|
TRUE | '"cats"' in ['"cats"', '"dogs"] Note the use of lazy quantifiers ? makes the regular expression match the shortest string possible in the text, causing '"cats"', '"dogs"' to be extracted as 2 separate tokens in the text. |
Examples
The following examples demonstrates the use of text analysis with customization options to create search indexes, extract tokens, and return search results.
LOG_ANALYZER
with NFKC ICU normalization and stop words
The following example configures LOG_ANALYZER
options with NFKC ICU
normalization and stop words. The example assumes the following data table with
data already populated:
CREATE TABLE dataset.data_table( text_data STRING );
To create a search index with NFKC ICU normalization and a list of stop words,
create a JSON-formatted string in the analyzer_options
option of the CREATE
SEARCH INDEX
DDL
statement.
For a complete list of options available in when creating a search index with
the LOG_ANALYZER
, see
LOG_ANALYZER
.
For this example, our stop words are "the", "of", "and", "for"
.
CREATE OR REPLACE SEARCH INDEX `my_index` ON `dataset.data_table`(ALL COLUMNS) OPTIONS( analyzer='PATTERN_ANALYZER', analyzer_options= '''{ "token_filters": [ { "normalizer": { "mode": "ICU_NORMALIZE", "icu_normalize_mode": "NFKC", "icu_case_folding": true } }, { "stop_words": ["the", "of", "and", "for"] } ] }''');
Given the previous example, the following table describes the token extraction
for various values of text_data
. Note that in this document the double
question mark character (⁇) has been italicized to differentiate between
two question marks (??):
Data Text | Tokens for index | Explanation |
---|---|---|
The Quick Brown Fox | ["quick", "brown", "fox"] | LOG_ANALYZER tokenization produces the tokens ["The", "Quick", "Brown", "Fox"]. Next, ICU normalization with icu_case_folding = true lower cases the tokens to produce ["the", "quick", "brown", "fox"]Finally, the stop words filter removes "the" from the list. |
The Ⓠuick Ⓑrown Ⓕox | ["quick", "brown", "fox"] | LOG_ANALYZER tokenization produces the tokens ["The", "Ⓠuick", "Ⓑrown", "Ⓕox"]. Next, NFKC ICU normalization with icu_case_folding = true lower cases the tokens to produce ["the", "quick", "brown", "fox"]Finally, the stop words filter removes "the" from the list. |
Ⓠuick⁇Ⓕox | ["quick??fox"] | LOG_ANALYZER tokenization produces the tokens ["The", "Ⓠuick⁇Ⓕox"]. Next, NFKC ICU normalization with icu_case_folding = true lower cases the tokens to produce ["quick??fox"]. Notice that the double question mark unicode has been normalized into 2 question mark ASCII characters.Finally, the stop words filter does nothing because none of the tokens are in the filter list. |
Now that the search index has been created, you can use the SEARCH
function to search the
table using the same analyzer configurations specified in the search index. Note
that if the analyzer configurations in the SEARCH
function don't match those
of the search index, the search index won't be used. Use the following query:
SELECT SEARCH( analyzer => 'LOG_ANALYZER', analyzer_options => '''{ "token_filters": [ { "normalizer": { "mode": "ICU_NORMALIZE", "icu_normalize_mode": "NFKC", "icu_case_folding": true } }, { "stop_words": ["the", "of", "and", "for"] } ] }''')
Replace the following:
search_query
: The text you want to search for.
The following
table demonstrates various results based on different search text and different
values of search_query
:
text_data | search_query |
Result | Explanation |
---|---|---|---|
The Quick Brown Fox | "Ⓠuick" |
TRUE |
The final list of tokens extracted
from the text is ["quick", "brown", "fox"]. The final list of tokens extracted from the text query is ["quick"]. The list query tokens can all be found in the text tokens. |
The Ⓠuick Ⓑrown Ⓕox | "quick" |
TRUE |
The final list of tokens extracted from the text is ["quick", "brown", "fox"]. The final list of tokens extracted from the text query is ["quick"]. The list query tokens can all be found in the text tokens. |
Ⓠuick⁇Ⓕox | "quick" |
FALSE |
The final list of tokens extracted from the text is ["quick??fox"]. The final list of tokens extracted from the text query is ["quick"]. "quick" is not in the list of tokens from the text. |
Ⓠuick⁇Ⓕox | "quick⁇fox" |
TRUE |
The final list of tokens extracted from the text is ["quick??fox"]. The final list of tokens extracted from the text query is ["quick??fox"]. "quick??fox" is in the list of tokens from the text. |
Ⓠuick⁇Ⓕox | "`quick⁇fox`" |
FALSE |
In LOG_ANALYZER , backtick requires exact text match. |
PATTERN_ANALYZER
for IPv4 search with stop words
The following example configures the PATTERN_ANALYZER
text analyzer to search for a specific pattern while filtering certain stop words. In this example, the pattern matches an IPv4 address and ignores the localhost value (127.0.0.1
).
This example assumes that the following table is populated with data:
CREATE TABLE dataset.data_table( text_data STRING );
To create a search index the pattern
option and a list of stop words, create a
JSON-formatted string in the analyzer_options
option of the CREATE SEARCH
INDEX
DDL
statement.
For a complete list of options available in when creating a search index with
the PATTERN_ANALYZER
, see
PATTERN_ANALYZER
.
For this example, our stop words are the localhost address,
127.0.0.1
.
CREATE SEARCH INDEX my_index ON dataset.data_table(text_data) OPTIONS (analyzer = 'PATTERN_ANALYZER', analyzer_options = '''{ "patterns": [ "(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)[.]){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)" ], "token_filters": [ { "stop_words": [ "127.0.0.1" ] } ] }''' );
When using regular expressions with analyzer_options
, include three
leading \
symbols to properly escape regular expressions that include a
\
symbol, such as \d
or \b
.
The following table describes the tokenization options for various values of text_data
Data Text | Tokens for index | Explanation |
---|---|---|
abc192.168.1.1def 172.217.20.142 | ["192.168.1.1", "172.217.20.142"] | The IPv4 patterns capture the IPv4 addresses even if there's no space between the address and the text. |
104.24.12.10abc 127.0.0.1 | ["104.24.12.10"] | "127.0.0.1" is filtered out since it's in the list of stop words. |
Now that the search index has been created, you can use the SEARCH
function to search the
table based on the tokenization specified in analyzer_options
. Use the
following query:
SELECT SEARCH(dataset.data_table.text_data "search_data", analyzer => 'PATTERN_ANALYZER', analyzer_options => '''{ "patterns": [ "(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)[.]){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)" ], "token_filters": [ { "stop_words": [ "127.0.0.1" ] } ] }''' );
Replace the following:
search_query
: The text you want to search for.
The following
table demonstrates various results based on different search text and different
values of search_query
:
text_data | search_query |
Result | Explanation |
---|---|---|---|
128.0.0.2 | "127.0.0.1" | ERROR | No search token in query. The query goes through the text analyzer, which filters out the "127.0.0.1" token. |
abc192.168.1.1def 172.217.20.142 | "192.168.1.1abc" | TRUE | The list of tokens extracted from the query is ["192.168.1.1"]. The list of tokens extracted from text is ["192.168.1.1", "172.217.20.142"]. |
abc192.168.1.1def 172.217.20.142 | "`192.168.1.1`" | TRUE | The list of tokens extracted from the query is ["192.168.1.1"]. The list of tokens extracted from text is ["192.168.1.1", "172.217.20.142"]. Note that backticks are treated as regular characters for PATTERN_ANALYZER. |
What's next
- For an overview of search index use cases, pricing, required permissions, and limitations, see the Introduction to search in BigQuery.
- For information about efficient searching of indexed columns, see Search with an index.