Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
M
mariadb
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
mariadb
Commits
4a5ca107
Commit
4a5ca107
authored
Nov 28, 2001
by
serg@serg.mysql.com
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
boolean fulltext search without an index
parent
79258183
Changes
13
Hide whitespace changes
Inline
Side-by-side
Showing
13 changed files
with
113 additions
and
118 deletions
+113
-118
Docs/manual.texi
Docs/manual.texi
+10
-6
include/ft_global.h
include/ft_global.h
+1
-1
include/my_base.h
include/my_base.h
+1
-0
myisam/ft_boolean_search.c
myisam/ft_boolean_search.c
+15
-7
myisam/ft_dump.c
myisam/ft_dump.c
+10
-2
myisam/ft_nlq_search.c
myisam/ft_nlq_search.c
+8
-4
myisam/ft_parser.c
myisam/ft_parser.c
+3
-3
myisam/ft_update.c
myisam/ft_update.c
+3
-3
myisam/ftdefs.h
myisam/ftdefs.h
+3
-3
mysql-test/t/fulltext.test
mysql-test/t/fulltext.test
+13
-0
sql/item_func.cc
sql/item_func.cc
+34
-70
sql/item_func.h
sql/item_func.h
+10
-17
sql/sql_select.cc
sql/sql_select.cc
+2
-2
No files found.
Docs/manual.texi
View file @
4a5ca107
...
@@ -29167,8 +29167,6 @@ mysql> select STRCMP('text', 'text');
...
@@ -29167,8 +29167,6 @@ mysql> select STRCMP('text', 'text');
relevance - similarity measure between the text in columns
relevance - similarity measure between the text in columns
@code{(col1,col2,...)} and the query @code{expr}. Relevance is a
@code{(col1,col2,...)} and the query @code{expr}. Relevance is a
positive floating-point number. Zero relevance means no similarity.
positive floating-point number. Zero relevance means no similarity.
For @code{MATCH ... AGAINST()} to work, a @strong{FULLTEXT} index
must be created first. @xref{CREATE TABLE, , @code{CREATE TABLE}}.
@code{MATCH ... AGAINST()} is available in MySQL version
@code{MATCH ... AGAINST()} is available in MySQL version
3.23.23 or later. @code{IN BOOLEAN MODE} extension was added in version
3.23.23 or later. @code{IN BOOLEAN MODE} extension was added in version
4.0.1. For details and usage examples @pxref{Fulltext Search}.
4.0.1. For details and usage examples @pxref{Fulltext Search}.
...
@@ -33828,9 +33826,10 @@ mysql> SELECT * FROM articles WHERE MATCH (title,body) AGAINST (
...
@@ -33828,9 +33826,10 @@ mysql> SELECT * FROM articles WHERE MATCH (title,body) AGAINST (
This query retrieved all the rows that contain the word @code{MySQL}
This query retrieved all the rows that contain the word @code{MySQL}
(note: 50% threshold is gone), but does @strong{not} contain the word
(note: 50% threshold is gone), but does @strong{not} contain the word
@code{YourSQL}. Note that it does not auto-magically sort rows in
@code{YourSQL}. Note
,
that it does not auto-magically sort rows in
derceasing relevance order (the last row has the highest relevance,
derceasing relevance order (the last row has the highest relevance,
as it contains @code{MySQL} twice).
as it contains @code{MySQL} twice). Boolean fulltext search can also
work even without @code{FULLTEXT} index, but it would be @strong{slow}.
Boolean fulltext search supports the following operators:
Boolean fulltext search supports the following operators:
...
@@ -33890,10 +33889,12 @@ order), but rank ``gates to hell'' higher than ``bill gates''.
...
@@ -33890,10 +33889,12 @@ order), but rank ``gates to hell'' higher than ``bill gates''.
@itemize @bullet
@itemize @bullet
@item
@item
All parameters to the @code{MATCH} function must be columns from the
All parameters to the @code{MATCH} function must be columns from the
same table that is part of the same fulltext index.
same table that is part of the same fulltext index, unless this
@code{MATCH} is @code{IN BOOLEAN MODE}.
@item
@item
Column list between @code{MATCH} and @code{AGAINST} must match exactly
Column list between @code{MATCH} and @code{AGAINST} must match exactly
a column list in the @code{FULLTEXT} index definition.
a column list in the @code{FULLTEXT} index definition, unless this
@code{MATCH} is @code{IN BOOLEAN MODE}.
@item
@item
The argument to @code{AGAINST} must be a constant string.
The argument to @code{AGAINST} must be a constant string.
@end itemize
@end itemize
...
@@ -45853,6 +45854,9 @@ Our TODO section contains what we plan to have in 4.0. @xref{TODO MySQL 4.0}.
...
@@ -45853,6 +45854,9 @@ Our TODO section contains what we plan to have in 4.0. @xref{TODO MySQL 4.0}.
@itemize @bullet
@itemize @bullet
@item
@item
@code{MATCH ... AGAINST(... IN BOOLEAN MODE)} can now work
without @code{FULLTEXT} index.
@item
Added @file{myisam/ft_dump} utility for low-level inspection
Added @file{myisam/ft_dump} utility for low-level inspection
of @code{FULLTEXT} indexes.
of @code{FULLTEXT} indexes.
@item
@item
include/ft_global.h
View file @
4a5ca107
...
@@ -32,7 +32,7 @@ extern "C" {
...
@@ -32,7 +32,7 @@ extern "C" {
typedef
struct
st_ft_info
FT_INFO
;
typedef
struct
st_ft_info
FT_INFO
;
struct
_ft_vft
{
struct
_ft_vft
{
int
(
*
read_next
)(
FT_INFO
*
,
char
*
);
int
(
*
read_next
)(
FT_INFO
*
,
char
*
);
float
(
*
find_relevance
)(
FT_INFO
*
,
my_off_t
,
byte
*
);
float
(
*
find_relevance
)(
FT_INFO
*
,
byte
*
,
uint
);
void
(
*
close_search
)(
FT_INFO
*
);
void
(
*
close_search
)(
FT_INFO
*
);
float
(
*
get_relevance
)(
FT_INFO
*
);
float
(
*
get_relevance
)(
FT_INFO
*
);
void
(
*
reinit_search
)(
FT_INFO
*
);
void
(
*
reinit_search
)(
FT_INFO
*
);
...
...
include/my_base.h
View file @
4a5ca107
...
@@ -226,6 +226,7 @@ enum ha_base_keytype {
...
@@ -226,6 +226,7 @@ enum ha_base_keytype {
/* Other constants */
/* Other constants */
#define HA_NAMELEN 64
/* Max length of saved filename */
#define HA_NAMELEN 64
/* Max length of saved filename */
#define NO_SUCH_KEY ((uint)~0)
/* used as a key no. */
/* Intern constants in databases */
/* Intern constants in databases */
...
...
myisam/ft_boolean_search.c
View file @
4a5ca107
...
@@ -152,13 +152,16 @@ void _ftb_init_index_search(FT_INFO *ftb)
...
@@ -152,13 +152,16 @@ void _ftb_init_index_search(FT_INFO *ftb)
int
i
,
r
;
int
i
,
r
;
FTB_WORD
*
ftbw
;
FTB_WORD
*
ftbw
;
MI_INFO
*
info
=
ftb
->
info
;
MI_INFO
*
info
=
ftb
->
info
;
MI_KEYDEF
*
keyinfo
=
info
->
s
->
keyinfo
+
ftb
->
keynr
;
MI_KEYDEF
*
keyinfo
;
my_off_t
keyroot
=
info
->
s
->
state
.
key_root
[
ftb
->
keynr
]
;
my_off_t
keyroot
;
if
(
ftb
->
state
!=
READY
)
if
(
ftb
->
state
!=
READY
||
ftb
->
keynr
==
NO_SUCH_KEY
)
return
;
return
;
ftb
->
state
=
INDEX_SEARCH
;
ftb
->
state
=
INDEX_SEARCH
;
keyinfo
=
info
->
s
->
keyinfo
+
ftb
->
keynr
;
keyroot
=
info
->
s
->
state
.
key_root
[
ftb
->
keynr
];
for
(
i
=
ftb
->
queue
.
elements
;
i
;
i
--
)
for
(
i
=
ftb
->
queue
.
elements
;
i
;
i
--
)
{
{
ftbw
=
(
FTB_WORD
*
)(
ftb
->
queue
.
root
[
i
]);
ftbw
=
(
FTB_WORD
*
)(
ftb
->
queue
.
root
[
i
]);
...
@@ -352,14 +355,17 @@ int ft_boolean_read_next(FT_INFO *ftb, char *record)
...
@@ -352,14 +355,17 @@ int ft_boolean_read_next(FT_INFO *ftb, char *record)
return
my_errno
=
HA_ERR_END_OF_FILE
;
return
my_errno
=
HA_ERR_END_OF_FILE
;
}
}
float
ft_boolean_find_relevance
(
FT_INFO
*
ftb
,
my_off_t
docid
,
byte
*
record
)
float
ft_boolean_find_relevance
(
FT_INFO
*
ftb
,
byte
*
record
,
uint
length
)
{
{
TREE
ptree
;
TREE
ptree
;
FT_WORD
word
;
FT_WORD
word
;
FTB_WORD
*
ftbw
;
FTB_WORD
*
ftbw
;
FTB_EXPR
*
ftbe
;
FTB_EXPR
*
ftbe
;
uint
i
;
uint
i
;
my_off_t
docid
=
ftb
->
info
->
lastpos
;
if
(
docid
==
HA_POS_ERROR
)
return
-
2
.
0
;
if
(
ftb
->
state
==
READY
||
ftb
->
state
==
INDEX_DONE
)
if
(
ftb
->
state
==
READY
||
ftb
->
state
==
INDEX_DONE
)
{
{
for
(
i
=
1
;
i
<=
ftb
->
queue
.
elements
;
i
++
)
for
(
i
=
1
;
i
<=
ftb
->
queue
.
elements
;
i
++
)
...
@@ -382,11 +388,13 @@ float ft_boolean_find_relevance(FT_INFO *ftb, my_off_t docid, byte *record)
...
@@ -382,11 +388,13 @@ float ft_boolean_find_relevance(FT_INFO *ftb, my_off_t docid, byte *record)
ftb
->
state
=
SCAN
;
ftb
->
state
=
SCAN
;
}
}
else
if
(
ftb
->
state
!=
SCAN
)
else
if
(
ftb
->
state
!=
SCAN
)
return
-
2
.
0
;
return
-
3
.
0
;
bzero
(
&
ptree
,
sizeof
(
ptree
));
bzero
(
&
ptree
,
sizeof
(
ptree
));
if
(
_mi_ft_parse
(
&
ptree
,
ftb
->
info
,
ftb
->
keynr
,
record
))
if
((
ftb
->
keynr
==
NO_SUCH_KEY
)
return
-
3
.
0
;
?
ft_parse
(
&
ptree
,
record
,
length
)
:
_mi_ft_parse
(
&
ptree
,
ftb
->
info
,
ftb
->
keynr
,
record
))
return
-
4
.
0
;
for
(
i
=
1
;
i
<=
ftb
->
queue
.
elements
;
i
++
)
for
(
i
=
1
;
i
<=
ftb
->
queue
.
elements
;
i
++
)
{
{
...
...
myisam/ft_dump.c
View file @
4a5ca107
...
@@ -159,7 +159,7 @@ err:
...
@@ -159,7 +159,7 @@ err:
return
0
;
return
0
;
}
}
const
char
*
options
=
"dscv
e:
h"
;
const
char
*
options
=
"dscvh"
;
static
void
get_options
(
int
argc
,
char
*
argv
[])
static
void
get_options
(
int
argc
,
char
*
argv
[])
{
{
...
@@ -184,7 +184,15 @@ static void get_options(int argc, char *argv[])
...
@@ -184,7 +184,15 @@ static void get_options(int argc, char *argv[])
static
void
usage
(
char
*
argv
[])
static
void
usage
(
char
*
argv
[])
{
{
printf
(
"Use: %s [-%s] <table_name> <key_no>
\n
"
,
*
argv
,
options
);
printf
(
"
Use: %s [-%s] <table_name> <index_no>
-d dump index (incl. data offsets and word weights)
-s report global stats
-c calculate per-word stats (counts and global weights)
-v be verbose
-h this text
\n
"
,
*
argv
,
options
);
exit
(
1
);
exit
(
1
);
}
}
...
...
myisam/ft_nlq_search.c
View file @
4a5ca107
...
@@ -169,7 +169,7 @@ static int FT_DOC_cmp(FT_DOC *a, FT_DOC *b)
...
@@ -169,7 +169,7 @@ static int FT_DOC_cmp(FT_DOC *a, FT_DOC *b)
FT_INFO
*
ft_init_nlq_search
(
MI_INFO
*
info
,
uint
keynr
,
byte
*
query
,
FT_INFO
*
ft_init_nlq_search
(
MI_INFO
*
info
,
uint
keynr
,
byte
*
query
,
uint
query_len
,
my_bool
presort
)
uint
query_len
,
my_bool
presort
)
{
{
TREE
*
wtree
,
allocated_wtree
;
TREE
allocated_wtree
,
*
wtree
=&
allocated_wtree
;
ALL_IN_ONE
aio
;
ALL_IN_ONE
aio
;
FT_DOC
*
dptr
;
FT_DOC
*
dptr
;
FT_INFO
*
dlist
=
NULL
;
FT_INFO
*
dlist
=
NULL
;
...
@@ -193,7 +193,7 @@ FT_INFO *ft_init_nlq_search(MI_INFO *info, uint keynr, byte *query,
...
@@ -193,7 +193,7 @@ FT_INFO *ft_init_nlq_search(MI_INFO *info, uint keynr, byte *query,
init_tree
(
&
aio
.
dtree
,
0
,
0
,
sizeof
(
FT_SUPERDOC
),(
qsort_cmp2
)
&
FT_SUPERDOC_cmp
,
0
,
init_tree
(
&
aio
.
dtree
,
0
,
0
,
sizeof
(
FT_SUPERDOC
),(
qsort_cmp2
)
&
FT_SUPERDOC_cmp
,
0
,
NULL
,
NULL
);
NULL
,
NULL
);
if
(
!
(
wtree
=
ft_parse
(
&
allocated_wtree
,
query
,
query_len
)
))
if
(
ft_parse
(
&
allocated_wtree
,
query
,
query_len
))
goto
err
;
goto
err
;
if
(
tree_walk
(
wtree
,
(
tree_walk_action
)
&
walk_and_match
,
&
aio
,
if
(
tree_walk
(
wtree
,
(
tree_walk_action
)
&
walk_and_match
,
&
aio
,
...
@@ -247,11 +247,15 @@ int ft_nlq_read_next(FT_INFO *handler, char *record)
...
@@ -247,11 +247,15 @@ int ft_nlq_read_next(FT_INFO *handler, char *record)
return
my_errno
;
return
my_errno
;
}
}
float
ft_nlq_find_relevance
(
FT_INFO
*
handler
,
my_off_t
docid
,
float
ft_nlq_find_relevance
(
FT_INFO
*
handler
,
byte
*
record
__attribute__
((
unused
)))
byte
*
record
__attribute__
((
unused
))
,
uint
length
__attribute__
((
unused
))
)
{
{
int
a
,
b
,
c
;
int
a
,
b
,
c
;
FT_DOC
*
docs
=
handler
->
doc
;
FT_DOC
*
docs
=
handler
->
doc
;
my_off_t
docid
=
handler
->
info
->
lastpos
;
if
(
docid
==
HA_POS_ERROR
)
return
-
5
.
0
;
/* Assuming docs[] is sorted by dpos... */
/* Assuming docs[] is sorted by dpos... */
...
...
myisam/ft_parser.c
View file @
4a5ca107
...
@@ -206,7 +206,7 @@ byte ft_simple_get_word(byte **start, byte *end, FT_WORD *word)
...
@@ -206,7 +206,7 @@ byte ft_simple_get_word(byte **start, byte *end, FT_WORD *word)
return
0
;
return
0
;
}
}
TREE
*
ft_parse
(
TREE
*
wtree
,
byte
*
doc
,
int
doclen
)
int
ft_parse
(
TREE
*
wtree
,
byte
*
doc
,
int
doclen
)
{
{
byte
*
end
=
doc
+
doclen
;
byte
*
end
=
doc
+
doclen
;
FT_WORD
w
;
FT_WORD
w
;
...
@@ -221,10 +221,10 @@ TREE * ft_parse(TREE *wtree, byte *doc, int doclen)
...
@@ -221,10 +221,10 @@ TREE * ft_parse(TREE *wtree, byte *doc, int doclen)
if
(
!
tree_insert
(
wtree
,
&
w
,
0
))
if
(
!
tree_insert
(
wtree
,
&
w
,
0
))
goto
err
;
goto
err
;
}
}
return
wtree
;
return
0
;
err:
err:
delete_tree
(
wtree
);
delete_tree
(
wtree
);
return
NULL
;
return
1
;
}
}
myisam/ft_update.c
View file @
4a5ca107
...
@@ -28,7 +28,7 @@
...
@@ -28,7 +28,7 @@
/**************************************************************/
/**************************************************************/
/* parses a document i.e. calls
_mi_
ft_parse for every keyseg */
/* parses a document i.e. calls ft_parse for every keyseg */
uint
_mi_ft_parse
(
TREE
*
parsed
,
MI_INFO
*
info
,
uint
keynr
,
const
byte
*
record
)
uint
_mi_ft_parse
(
TREE
*
parsed
,
MI_INFO
*
info
,
uint
keynr
,
const
byte
*
record
)
{
{
byte
*
pos
;
byte
*
pos
;
...
@@ -57,11 +57,11 @@ uint _mi_ft_parse(TREE *parsed, MI_INFO *info, uint keynr, const byte *record)
...
@@ -57,11 +57,11 @@ uint _mi_ft_parse(TREE *parsed, MI_INFO *info, uint keynr, const byte *record)
}
}
else
else
len
=
keyseg
->
length
;
len
=
keyseg
->
length
;
if
(
!
(
ft_parse
(
parsed
,
pos
,
len
)
))
if
(
ft_parse
(
parsed
,
pos
,
len
))
return
1
;
return
1
;
}
}
/* Handle the case where all columns are NULL */
/* Handle the case where all columns are NULL */
if
(
!
is_tree_inited
(
parsed
)
&&
!
(
ft_parse
(
parsed
,
(
byte
*
)
""
,
0
)
))
if
(
!
is_tree_inited
(
parsed
)
&&
ft_parse
(
parsed
,
(
byte
*
)
""
,
0
))
return
1
;
return
1
;
else
else
return
0
;
return
0
;
...
...
myisam/ftdefs.h
View file @
4a5ca107
...
@@ -120,14 +120,14 @@ uint _ft_make_key(MI_INFO *, uint , byte *, FT_WORD *, my_off_t);
...
@@ -120,14 +120,14 @@ uint _ft_make_key(MI_INFO *, uint , byte *, FT_WORD *, my_off_t);
byte
ft_get_word
(
byte
**
,
byte
*
,
FT_WORD
*
,
FTB_PARAM
*
);
byte
ft_get_word
(
byte
**
,
byte
*
,
FT_WORD
*
,
FTB_PARAM
*
);
byte
ft_simple_get_word
(
byte
**
,
byte
*
,
FT_WORD
*
);
byte
ft_simple_get_word
(
byte
**
,
byte
*
,
FT_WORD
*
);
TREE
*
ft_parse
(
TREE
*
,
byte
*
,
int
);
int
ft_parse
(
TREE
*
,
byte
*
,
int
);
FT_WORD
*
ft_linearize
(
/*MI_INFO *, uint, byte *, */
TREE
*
);
FT_WORD
*
ft_linearize
(
/*MI_INFO *, uint, byte *, */
TREE
*
);
FT_WORD
*
_mi_ft_parserecord
(
MI_INFO
*
,
uint
,
byte
*
,
const
byte
*
);
FT_WORD
*
_mi_ft_parserecord
(
MI_INFO
*
,
uint
,
byte
*
,
const
byte
*
);
const
struct
_ft_vft
_ft_vft_nlq
;
const
struct
_ft_vft
_ft_vft_nlq
;
FT_INFO
*
ft_init_nlq_search
(
MI_INFO
*
,
uint
,
byte
*
,
uint
,
my_bool
);
FT_INFO
*
ft_init_nlq_search
(
MI_INFO
*
,
uint
,
byte
*
,
uint
,
my_bool
);
int
ft_nlq_read_next
(
FT_INFO
*
,
char
*
);
int
ft_nlq_read_next
(
FT_INFO
*
,
char
*
);
float
ft_nlq_find_relevance
(
FT_INFO
*
,
my_off_t
,
byte
*
);
float
ft_nlq_find_relevance
(
FT_INFO
*
,
byte
*
,
uint
);
void
ft_nlq_close_search
(
FT_INFO
*
);
void
ft_nlq_close_search
(
FT_INFO
*
);
float
ft_nlq_get_relevance
(
FT_INFO
*
);
float
ft_nlq_get_relevance
(
FT_INFO
*
);
my_off_t
ft_nlq_get_docid
(
FT_INFO
*
);
my_off_t
ft_nlq_get_docid
(
FT_INFO
*
);
...
@@ -136,7 +136,7 @@ void ft_nlq_reinit_search(FT_INFO *);
...
@@ -136,7 +136,7 @@ void ft_nlq_reinit_search(FT_INFO *);
const
struct
_ft_vft
_ft_vft_boolean
;
const
struct
_ft_vft
_ft_vft_boolean
;
FT_INFO
*
ft_init_boolean_search
(
MI_INFO
*
,
uint
,
byte
*
,
uint
,
my_bool
);
FT_INFO
*
ft_init_boolean_search
(
MI_INFO
*
,
uint
,
byte
*
,
uint
,
my_bool
);
int
ft_boolean_read_next
(
FT_INFO
*
,
char
*
);
int
ft_boolean_read_next
(
FT_INFO
*
,
char
*
);
float
ft_boolean_find_relevance
(
FT_INFO
*
,
my_off_t
,
byte
*
);
float
ft_boolean_find_relevance
(
FT_INFO
*
,
byte
*
,
uint
);
void
ft_boolean_close_search
(
FT_INFO
*
);
void
ft_boolean_close_search
(
FT_INFO
*
);
float
ft_boolean_get_relevance
(
FT_INFO
*
);
float
ft_boolean_get_relevance
(
FT_INFO
*
);
my_off_t
ft_boolean_get_docid
(
FT_INFO
*
);
my_off_t
ft_boolean_get_docid
(
FT_INFO
*
);
...
...
mysql-test/t/fulltext.test
View file @
4a5ca107
...
@@ -10,9 +10,15 @@ INSERT INTO t1 VALUES('MySQL has now support', 'for full-text search'),
...
@@ -10,9 +10,15 @@ INSERT INTO t1 VALUES('MySQL has now support', 'for full-text search'),
(
'Only MyISAM tables'
,
'support collections'
),
(
'Only MyISAM tables'
,
'support collections'
),
(
'Function MATCH ... AGAINST()'
,
'is used to do a search'
),
(
'Function MATCH ... AGAINST()'
,
'is used to do a search'
),
(
'Full-text search in MySQL'
,
'implements vector space model'
);
(
'Full-text search in MySQL'
,
'implements vector space model'
);
# nl search
select
*
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"collections"
);
select
*
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"collections"
);
select
*
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"indexes"
);
select
*
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"indexes"
);
select
*
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"indexes collections"
);
select
*
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"indexes collections"
);
# boolean search
select
*
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"support -collections"
IN
BOOLEAN
MODE
);
select
*
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"support -collections"
IN
BOOLEAN
MODE
);
select
*
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"support collections"
IN
BOOLEAN
MODE
);
select
*
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"support collections"
IN
BOOLEAN
MODE
);
select
*
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"support +collections"
IN
BOOLEAN
MODE
);
select
*
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"support +collections"
IN
BOOLEAN
MODE
);
...
@@ -22,6 +28,13 @@ select * from t1 where MATCH(a,b) AGAINST("+search" IN BOOLEAN MODE);
...
@@ -22,6 +28,13 @@ select * from t1 where MATCH(a,b) AGAINST("+search" IN BOOLEAN MODE);
select
*
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"+search +(support vector)"
IN
BOOLEAN
MODE
);
select
*
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"+search +(support vector)"
IN
BOOLEAN
MODE
);
select
*
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"+search -(support vector)"
IN
BOOLEAN
MODE
);
select
*
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"+search -(support vector)"
IN
BOOLEAN
MODE
);
select
*
,
MATCH
(
a
,
b
)
AGAINST
(
"support collections"
IN
BOOLEAN
MODE
)
as
x
from
t1
;
select
*
,
MATCH
(
a
,
b
)
AGAINST
(
"support collections"
IN
BOOLEAN
MODE
)
as
x
from
t1
;
# boolean w/o index:
select
*
from
t1
where
MATCH
a
AGAINST
(
"search"
IN
BOOLEAN
MODE
);
#update/delete with fulltext index
delete
from
t1
where
a
like
"MySQL%"
;
delete
from
t1
where
a
like
"MySQL%"
;
update
t1
set
a
=
'some test foobar'
where
MATCH
a
,
b
AGAINST
(
'model'
);
update
t1
set
a
=
'some test foobar'
where
MATCH
a
,
b
AGAINST
(
'model'
);
delete
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"indexes"
);
delete
from
t1
where
MATCH
(
a
,
b
)
AGAINST
(
"indexes"
);
...
...
sql/item_func.cc
View file @
4a5ca107
...
@@ -2004,6 +2004,9 @@ void Item_func_match::init_search(bool no_order)
...
@@ -2004,6 +2004,9 @@ void Item_func_match::init_search(bool no_order)
return
;
return
;
}
}
if
(
key
==
NO_SUCH_KEY
)
concat
=
new
Item_func_concat_ws
(
new
Item_string
(
" "
,
1
),
fields
);
String
*
ft_tmp
=
0
;
String
*
ft_tmp
=
0
;
char
tmp1
[
FT_QUERY_MAXLEN
];
char
tmp1
[
FT_QUERY_MAXLEN
];
String
tmp2
(
tmp1
,
sizeof
(
tmp1
));
String
tmp2
(
tmp1
,
sizeof
(
tmp1
));
...
@@ -2015,7 +2018,8 @@ void Item_func_match::init_search(bool no_order)
...
@@ -2015,7 +2018,8 @@ void Item_func_match::init_search(bool no_order)
tmp2
.
set
(
""
,
0
);
tmp2
.
set
(
""
,
0
);
}
}
ft_handler_init
(
ft_tmp
->
ptr
(),
ft_tmp
->
length
(),
join_key
&&
!
no_order
);
ft_handler
=
table
->
file
->
ft_init_ext
(
mode
,
key
,
ft_tmp
->
ptr
(),
ft_tmp
->
length
(),
join_key
&&
!
no_order
);
if
(
join_key
)
if
(
join_key
)
{
{
...
@@ -2032,12 +2036,11 @@ bool Item_func_match::fix_fields(THD *thd,struct st_table_list *tlist)
...
@@ -2032,12 +2036,11 @@ bool Item_func_match::fix_fields(THD *thd,struct st_table_list *tlist)
maybe_null
=
1
;
maybe_null
=
1
;
join_key
=
0
;
join_key
=
0
;
/* Serg:
/* const_item is assumed in quite a bit of places, so it would be difficult
I'd rather say now that const_item is assumed in quite a bit of
to remove; If it would ever to be removed, this should include
places, so it would be difficult to remove; If it would ever to be
modifications to find_best and auto_close as complement to auto_init code
removed, this should include modifications to find_best and auto_close
above.
as complement to auto_init code above.
*/
*/
if
(
Item_func
::
fix_fields
(
thd
,
tlist
)
||
!
const_item
())
if
(
Item_func
::
fix_fields
(
thd
,
tlist
)
||
!
const_item
())
{
{
my_error
(
ER_WRONG_ARGUMENTS
,
MYF
(
0
),
"AGAINST"
);
my_error
(
ER_WRONG_ARGUMENTS
,
MYF
(
0
),
"AGAINST"
);
...
@@ -2051,21 +2054,20 @@ bool Item_func_match::fix_fields(THD *thd,struct st_table_list *tlist)
...
@@ -2051,21 +2054,20 @@ bool Item_func_match::fix_fields(THD *thd,struct st_table_list *tlist)
if
(
item
->
type
()
==
Item
::
REF_ITEM
)
if
(
item
->
type
()
==
Item
::
REF_ITEM
)
li
.
replace
(
item
=
*
((
Item_ref
*
)
item
)
->
ref
);
li
.
replace
(
item
=
*
((
Item_ref
*
)
item
)
->
ref
);
if
(
item
->
type
()
!=
Item
::
FIELD_ITEM
||
!
item
->
used_tables
())
if
(
item
->
type
()
!=
Item
::
FIELD_ITEM
||
!
item
->
used_tables
())
{
key
=
NO_SUCH_KEY
;
my_error
(
ER_WRONG_ARGUMENTS
,
MYF
(
0
),
"MATCH"
);
return
1
;
}
used_tables_cache
|=
item
->
used_tables
();
used_tables_cache
|=
item
->
used_tables
();
}
}
/* check that all columns come from the same table */
/* check that all columns come from the same table */
if
(
count_bits
(
used_tables_cache
)
!=
1
)
if
(
count_bits
(
used_tables_cache
)
!=
1
)
key
=
NO_SUCH_KEY
;
const_item_cache
=
0
;
table
=
((
Item_field
*
)
fields
.
head
())
->
field
->
table
;
record
=
table
->
record
[
0
];
if
(
key
==
NO_SUCH_KEY
&&
mode
!=
FT_BOOL
)
{
{
my_error
(
ER_WRONG_ARGUMENTS
,
MYF
(
0
),
"MATCH"
);
my_error
(
ER_WRONG_ARGUMENTS
,
MYF
(
0
),
"MATCH"
);
return
1
;
return
1
;
}
}
const_item_cache
=
0
;
table
=
((
Item_field
*
)
fields
.
head
())
->
field
->
table
;
record
=
table
->
record
[
0
];
return
0
;
return
0
;
}
}
...
@@ -2074,6 +2076,10 @@ bool Item_func_match::fix_index()
...
@@ -2074,6 +2076,10 @@ bool Item_func_match::fix_index()
List_iterator_fast
<
Item
>
li
(
fields
);
List_iterator_fast
<
Item
>
li
(
fields
);
Item_field
*
item
;
Item_field
*
item
;
uint
ft_to_key
[
MAX_KEY
],
ft_cnt
[
MAX_KEY
],
fts
=
0
,
key
;
uint
ft_to_key
[
MAX_KEY
],
ft_cnt
[
MAX_KEY
],
fts
=
0
,
key
;
uint
max_cnt
=
0
,
mkeys
=
0
;
if
(
this
->
key
==
NO_SUCH_KEY
)
return
0
;
for
(
key
=
0
;
key
<
table
->
keys
;
key
++
)
for
(
key
=
0
;
key
<
table
->
keys
;
key
++
)
{
{
...
@@ -2087,11 +2093,7 @@ bool Item_func_match::fix_index()
...
@@ -2087,11 +2093,7 @@ bool Item_func_match::fix_index()
}
}
if
(
!
fts
)
if
(
!
fts
)
{
goto
err
;
my_printf_error
(
ER_FT_MATCHING_KEY_NOT_FOUND
,
ER
(
ER_FT_MATCHING_KEY_NOT_FOUND
),
MYF
(
0
));
return
1
;
}
while
((
item
=
(
Item_field
*
)(
li
++
)))
while
((
item
=
(
Item_field
*
)(
li
++
)))
{
{
...
@@ -2108,7 +2110,6 @@ bool Item_func_match::fix_index()
...
@@ -2108,7 +2110,6 @@ bool Item_func_match::fix_index()
}
}
}
}
uint
max_cnt
=
0
,
mkeys
=
0
;
for
(
key
=
0
;
key
<
fts
;
key
++
)
for
(
key
=
0
;
key
<
fts
;
key
++
)
{
{
if
(
ft_cnt
[
key
]
>
max_cnt
)
if
(
ft_cnt
[
key
]
>
max_cnt
)
...
@@ -2139,6 +2140,12 @@ bool Item_func_match::fix_index()
...
@@ -2139,6 +2140,12 @@ bool Item_func_match::fix_index()
return
0
;
return
0
;
}
}
err:
if
(
mode
==
FT_BOOL
)
{
this
->
key
=
NO_SUCH_KEY
;
return
0
;
}
my_printf_error
(
ER_FT_MATCHING_KEY_NOT_FOUND
,
my_printf_error
(
ER_FT_MATCHING_KEY_NOT_FOUND
,
ER
(
ER_FT_MATCHING_KEY_NOT_FOUND
),
MYF
(
0
));
ER
(
ER_FT_MATCHING_KEY_NOT_FOUND
),
MYF
(
0
));
return
1
;
return
1
;
...
@@ -2174,61 +2181,18 @@ double Item_func_match::val()
...
@@ -2174,61 +2181,18 @@ double Item_func_match::val()
join_key
=
0
;
join_key
=
0
;
}
}
my_off_t
docid
=
table
->
file
->
row_position
();
if
(
key
==
NO_SUCH_KEY
)
if
((
null_value
=
(
docid
==
HA_OFFSET_ERROR
)))
return
0.0
;
else
return
ft_handler
->
please
->
find_relevance
(
ft_handler
,
docid
,
record
);
}
#if 0
double Item_func_match_nl::val()
{
if (ft_handler==NULL)
init_search(1);
if ((null_value= (ft_handler==NULL)))
return 0.0;
if (join_key)
{
{
if (table->file->ft_handler)
String
*
a
=
concat
->
val_str
(
&
value
);
return ft_handler->please->get_relevance(ft_handler);
if
(
null_value
=
(
a
==
0
))
return
0
;
join_key=0;
return
ft_handler
->
please
->
find_relevance
(
ft_handler
,
(
byte
*
)
a
->
ptr
(),
a
->
length
());
}
}
my_off_t docid=table->file->row_position();
if ((null_value=(docid==HA_OFFSET_ERROR)))
return 0.0;
else
else
return ft_handler->please->find_relevance(ft_handler,
docid, record
);
return
ft_handler
->
please
->
find_relevance
(
ft_handler
,
record
,
0
);
}
}
double Item_func_match_bool::val()
{
if (ft_handler==NULL)
init_search(1);
if ((null_value= (ft_handler==NULL)))
return 0.0;
if (join_key)
{
if (table->file->ft_handler)
return ft_handler->please->get_relevance(ft_handler);
join_key=0;
}
return ft_handler->please->find_relevance(ft_handler, docid, record);
//null_value=1;
//return -1.0;
}
#endif
/***************************************************************************
/***************************************************************************
System variables
System variables
This has to be recoded after we get more than 3 system variables
This has to be recoded after we get more than 3 system variables
...
...
sql/item_func.h
View file @
4a5ca107
...
@@ -862,15 +862,18 @@ class Item_func_match :public Item_real_func
...
@@ -862,15 +862,18 @@ class Item_func_match :public Item_real_func
{
{
public:
public:
List
<
Item
>
fields
;
List
<
Item
>
fields
;
Item
*
concat
;
String
value
;
TABLE
*
table
;
TABLE
*
table
;
uint
key
;
uint
key
,
mode
;
bool
join_key
;
bool
join_key
;
Item_func_match
*
master
;
Item_func_match
*
master
;
FT_INFO
*
ft_handler
;
FT_INFO
*
ft_handler
;
byte
*
record
;
byte
*
record
;
Item_func_match
(
List
<
Item
>
&
a
,
Item
*
b
)
:
Item_real_func
(
b
),
Item_func_match
(
List
<
Item
>
&
a
,
Item
*
b
)
:
Item_real_func
(
b
),
fields
(
a
),
table
(
0
),
join_key
(
0
),
master
(
0
),
ft_handler
(
0
)
{}
fields
(
a
),
table
(
0
),
join_key
(
0
),
master
(
0
),
ft_handler
(
0
),
key
(
0
),
concat
(
0
)
{}
~
Item_func_match
()
~
Item_func_match
()
{
{
if
(
!
master
&&
ft_handler
)
if
(
!
master
&&
ft_handler
)
...
@@ -880,8 +883,8 @@ public:
...
@@ -880,8 +883,8 @@ public:
if
(
join_key
)
if
(
join_key
)
table
->
file
->
ft_handler
=
0
;
table
->
file
->
ft_handler
=
0
;
}
}
if
(
concat
)
delete
concat
;
}
}
virtual
int
ft_handler_init
(
const
byte
*
key
,
uint
keylen
,
bool
presort
)
=
0
;
enum
Functype
functype
()
const
{
return
FT_FUNC
;
}
enum
Functype
functype
()
const
{
return
FT_FUNC
;
}
void
update_used_tables
()
{}
void
update_used_tables
()
{}
bool
fix_fields
(
THD
*
thd
,
struct
st_table_list
*
tlist
);
bool
fix_fields
(
THD
*
thd
,
struct
st_table_list
*
tlist
);
...
@@ -896,26 +899,16 @@ public:
...
@@ -896,26 +899,16 @@ public:
class
Item_func_match_nl
:
public
Item_func_match
class
Item_func_match_nl
:
public
Item_func_match
{
{
public:
public:
Item_func_match_nl
(
List
<
Item
>
&
a
,
Item
*
b
)
:
Item_func_match
(
a
,
b
)
{}
Item_func_match_nl
(
List
<
Item
>
&
a
,
Item
*
b
)
:
Item_func_match
(
a
,
b
)
{
mode
=
FT_NL
;
}
const
char
*
func_name
()
const
{
return
"match_nl"
;
}
const
char
*
func_name
()
const
{
return
"match_nl"
;
}
// double val();
int
ft_handler_init
(
const
byte
*
query
,
uint
querylen
,
bool
presort
)
{
ft_handler
=
table
->
file
->
ft_init_ext
(
FT_NL
,
key
,
query
,
querylen
,
presort
);
return
0
;
}
};
};
class
Item_func_match_bool
:
public
Item_func_match
class
Item_func_match_bool
:
public
Item_func_match
{
{
public:
public:
Item_func_match_bool
(
List
<
Item
>
&
a
,
Item
*
b
)
:
Item_func_match
(
a
,
b
)
{}
Item_func_match_bool
(
List
<
Item
>
&
a
,
Item
*
b
)
:
Item_func_match
(
a
,
b
)
{
mode
=
FT_BOOL
;
}
const
char
*
func_name
()
const
{
return
"match_bool"
;
}
const
char
*
func_name
()
const
{
return
"match_bool"
;
}
// double val();
int
ft_handler_init
(
const
byte
*
query
,
uint
querylen
,
bool
presort
)
{
ft_handler
=
table
->
file
->
ft_init_ext
(
FT_BOOL
,
key
,
query
,
querylen
,
presort
);
return
0
;
}
};
};
sql/sql_select.cc
View file @
4a5ca107
...
@@ -1457,7 +1457,7 @@ add_ft_keys(DYNAMIC_ARRAY *keyuse_array,
...
@@ -1457,7 +1457,7 @@ add_ft_keys(DYNAMIC_ARRAY *keyuse_array,
{
{
Item
*
item
;
Item
*
item
;
/*
/*
I
,
(Sergei) too lazy to implement proper recursive descent here,
I
'm
(Sergei) too lazy to implement proper recursive descent here,
and anyway, nobody will use such a stupid queries
and anyway, nobody will use such a stupid queries
that will require it :-)
that will require it :-)
May be later...
May be later...
...
@@ -1474,7 +1474,7 @@ add_ft_keys(DYNAMIC_ARRAY *keyuse_array,
...
@@ -1474,7 +1474,7 @@ add_ft_keys(DYNAMIC_ARRAY *keyuse_array,
}
}
}
}
if
(
!
cond_func
)
if
(
!
cond_func
||
cond_func
->
key
==
NO_SUCH_KEY
)
return
;
return
;
KEYUSE
keyuse
;
KEYUSE
keyuse
;
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment