Files implementing UDFs must be compiled and installed on the
          host where the server runs. This process is described below
          for the example UDF file
          sql/udf_example.c that is included in the
          MySQL source distribution.
        
The immediately following instructions are for Unix. Instructions for Windows are given later in this section.
          The udf_example.c file contains the
          following functions:
        
              metaphon() returns a metaphon string of
              the string argument. This is something like a soundex
              string, but it is more tuned for English.
            
              myfunc_double() returns the sum of the
              ASCII values of the characters in its arguments, divided
              by the sum of the length of its arguments.
            
              myfunc_int() returns the sum of the
              length of its arguments.
            
              sequence([const int]) returns a
              sequence starting from the given number or 1 if no number
              has been given.
            
              lookup() returns the IP number for a
              host name.
            
              reverse_lookup() returns the host name
              for an IP number. The function may be called either with a
              single string argument of the form
              'xxx.xxx.xxx.xxx' or with four numbers.
            
A dynamically loadable file should be compiled as a sharable object file, using a command something like this:
shell> gcc -shared -o udf_example.so udf_example.c
          If you are using gcc with
          configure and libtool
          (which is how MySQL is configured), you should be able to
          create udf_example.so with a simpler
          command:
        
shell> make udf_example.la
          After you compile a shared object containing UDFs, you must
          install it and tell MySQL about it. Compiling a shared object
          from udf_example.c using
          gcc directly produces a file named
          udf_example.so. Compiling the shared
          object using make produces a file named
          something like udf_example.so.0.0.0 in
          the .libs directory (the exact name may
          vary from platform to platform). Copy the shared object to the
          server's plugin directory and name it
          udf_example.so. This directory is given
          by the value of the
          plugin_dir system variable.
        
This is a change in MySQL 5.1. For earlier versions of MySQL, the shared object can be located in any directory that is searched by your system's dynamic linker.
          On some systems, the ldconfig program that
          configures the dynamic linker does not recognize a shared
          object unless its name begins with lib. In
          this case you should rename a file such as
          udf_example.so to
          libudf_example.so.
        
On Windows, you can compile user-defined functions by using the following procedure:
You need to obtain the Bazaar source repository for MySQL 5.1. See Section 2.3.3, “Installing from the Development Source Tree”.
You must obtain the CMake build utility from http://www.cmake.org. (Version 2.4.2 or later is required).
              In the source repository, look in the
              sql directory. There are files named
              udf_example.def
              udf_example.c there. Copy both files
              from this directory to your working directory.
            
              Create a CMake makefile
              (CMakeLists.txt) with these contents:
            
PROJECT(udf_example)
# Path for MySQL include directory
INCLUDE_DIRECTORIES("c:/mysql/include")
ADD_DEFINITIONS("-DHAVE_DLOPEN")
ADD_LIBRARY(udf_example MODULE udf_example.c udf_example.def)
TARGET_LINK_LIBRARIES(udf_example wsock32)
Create the VC project and solution files:
cmake -G "<Generator>"
Invoking cmake --help shows you a list of valid Generators.
              Create udf_example.dll:
            
devenv udf_example.sln /build Release
After the shared object file has been installed, notify mysqld about the new functions with these statements:
mysql>CREATE FUNCTION metaphon RETURNS STRING SONAME 'udf_example.dll';mysql>CREATE FUNCTION myfunc_double RETURNS REAL SONAME 'udf_example.dll';mysql>CREATE FUNCTION myfunc_int RETURNS INTEGER SONAME 'udf_example.dll';mysql>CREATE FUNCTION sequence RETURNS INTEGER SONAME 'udf_example.dll';mysql>CREATE FUNCTION lookup RETURNS STRING SONAME 'udf_example.dll';mysql>CREATE FUNCTION reverse_lookup->RETURNS STRING SONAME 'udf_example.dll';mysql>CREATE AGGREGATE FUNCTION avgcost->RETURNS REAL SONAME 'udf_example.dll';
          Functions can be deleted using DROP
          FUNCTION:
        
mysql>DROP FUNCTION metaphon;mysql>DROP FUNCTION myfunc_double;mysql>DROP FUNCTION myfunc_int;mysql>DROP FUNCTION sequence;mysql>DROP FUNCTION lookup;mysql>DROP FUNCTION reverse_lookup;mysql>DROP FUNCTION avgcost;
          The CREATE FUNCTION and
          DROP FUNCTION statements update
          the func system table in the
          mysql database. The function's name, type
          and shared library name are saved in the table. You must have
          the INSERT and
          DELETE privileges for the
          mysql database to create and drop
          functions.
        
          You should not use CREATE
          FUNCTION to add a function that has previously been
          created. If you need to reinstall a function, you should
          remove it with DROP FUNCTION
          and then reinstall it with CREATE
          FUNCTION. You would need to do this, for example, if
          you recompile a new version of your function, so that
          mysqld gets the new version. Otherwise, the
          server continues to use the old version.
        
          An active function is one that has been loaded with
          CREATE FUNCTION and not removed
          with DROP FUNCTION. All active
          functions are reloaded each time the server starts, unless you
          start mysqld with the
          --skip-grant-tables option. In
          this case, UDF initialization is skipped and UDFs are
          unavailable.
        
If the new function will be referred to in statements that will be replicated to slave servers, you must ensure that every slave server also has the function available. Otherwise, replication will fail on the slaves when they attempt to invoke the function.


User Comments
Helpful hint for debugging: fprintf(stderr,"Hello world"); anything printed to stderr will go to the mysql error log.
%lld is the proper format for printing long longs.
After much time trying to track a problem on solaris(2.7, gcc 2.9-gnupro-99r1) I found that I needed -c -shared as compiler options. Almost everything worked without the -c, but strangely, any time I tried to use double quoted strings in the code they got turned into empty strings.
On Mac OS X 10.4 (Tiger), the following incantations were needed, in
order to build a "shared object" file.
export MACOSX_DEPLOYMENT_TARGET="10.4"
gcc -c myfunction.c
gcc -bundle -o myfunction.so myfunction.o -undefined dynamic_lookup
I wanted to install the UDF to compute the levenshtein distance between two strings.
I put here all the commands i used, from scratch (both on the server and my mind), to make it running.
I had several little problems that you can't really solve without knowing some linux tips.
This is different from what I saw in this doc.
Here is the command for a debian sarge stable server.
>apt-get source mysql-server
>locate udf_example.c
>cd /usr/share/doc/libmysqlclient14-dev/examples
>gunzip udf_example.cc.gz
>gcc -shared -o udf_example.so udf_example.cc
>locate mysql.h
>gcc -shared -o udf_example.so udf_example.cc -I/usr/include/mysql/
>ll
>mv udf_example.so /usr/lib/libudf_example.so
#at this step i'm happy because the example from mysql works (note that i had to add the -I for the path and change the file extension to .cc)
# now i download the package from the mysql udf repository
# http://empyrean.lib.ndsu.nodak.edu/~nem/mysql/
>wget http://empyrean.lib.ndsu.nodak.edu/~nem/mysql/udf/dludf.cgi?ckey=28
>ll
>tar -xzvf dludf.cgi\?ckey\=28
>gcc -shared -o libmysqllevenshtein.so mysqllevenshtein.cc -I/usr/include/mysql/
>mv libmysqllevenshtein.so /usr/lib
#Then i launch mysql
>mysql -uroot -pPASS
# in mysql
mysql> use DATABASE
Database changed
mysql> CREATE FUNCTION levenshtein RETURNS INT SONAME 'libmysqllevenshtein.so';
mysql> select levenshtein(w1.word,w2.word) as dist from word w1, word w2 where ETC........... order by dist asc limit 0,10;
Enjoy :-)
To compile mysqllevenshtein (as it happens, though I imagine this would be the case with other functions) on Ubuntu Dapper Drake, I needed to add the -fPIC compiler flag:
gcc -shared -o mysqllevenshtein.so mysqllevenshtein.cc \
-I/usr/include/mysql/ -fPIC
/afb
To clarify the Mac OS X example slightly: I use the following commands to build and install the example UDF on OS X 10.4. These should be executed in the directory where udf_example.c is stored.
# On OS X, use "-bundle" instead of "-shared",
# and specify where the mysql include files are
# (they should be in your main mysql directory).
gcc -bundle -o udf_example.so udf_example.c -I/usr/local/mysql/include
# MySQL will look for the shared library in /usr/lib, so put it there:
sudo cp udf_example.so /usr/lib
If you're still in trouble with the levenshtein function, here you can find a manual concerning the installation of the levenshtein group function: http://www.teamarbyte.de/levenshtein.html
It's in German, but I hope it helps anyway.
Add your own comment.