Save the planet. Save the trees. Save your data, man.
Project powered by:
|
MySQL Serializer
Did ya ever want to save your objects into a database? Or convert them
from whatever serializable format they live in, to a database? Now we
can... Using this add-on DLL we can convert any and all s11n-saved
objects and files to and from database entries. We can also save and
load our objects directly to/from a database, ignoring files
altogether. And, of course, s11n's API ensures that our client-side
code doesn't need to know if it's using a database server in Taiwan
or if it's saving to ~/foo.s11n.
ACHTUNG: This code is not yet ported to s11n versions newer than 1.0.x, and may
never be. If you actually use this add-on, let me know and i'll port it
to 1.1+ for you, otherwise i probably won't bother.
mysql_serializer ("MyS", for short)
Credits:
The s11n::io::mysql_serializer class is powered by a C++
library for mysql written by
Anders Hedström.
He's got an impressive set of utility libraries on his site.
Using MyS does not require his mysql library: this tree comes with a slightly hacked copy of his
code.
MyS can load and save objects from and to a mysql database, and works
essentially just like any other s11n serializer:
client-transparently. That is, the client never needs to know if his
data is going to an XML file, an s11n-specific data format, or
a database.
This Serializer is most interesting for following points:
- The core library is 100% ignorant of it, and can load it by being given
it's class name.
- It proves that clients can provide their own i/o handlers
without touching the core library. (Previously this was pure theory.)
A client-written Serializer becomes a full-rights player in the framework,
just like the Serializers shipped with the library.
- It demonstrates that the previous point can be accomplished with relatively
little effort. i spent about 8 hours coding it and touching it up, including
the learning curve for the mysql-related code, and tweaking s11nconvert
and the core lib to be able to import DLLs and arbitrary Serializers this way.
- It demonstrates that non-file-based Serializers are a realistic option,
and are treated with the same API transparency as their file-based cousins.
-
Demonstrates the ability to chain Serializers together to do more complex work.
This one, e.g., when serializing to a STREAM it saves a record to the db and
saves proxying information to the stream using a different Serializer
(i.e., one that natively supports streams). That proxying info can later be loaded
just like any s11n data, but it will direct MyS to load the db object from the database.
Source code...
First, please see the Big Fat Caveats detailed below!
(i'll wait...)
Then visit the downloads page.
The 2004.08.15 release of MyS will work with s11n 0.9.6 but does not
have STREAM i/o support. The 2004.08.16+ release supports both files
and streams but requires s11n 0.9.7 (which won't be released until 17 Aug).
Big Fat Caveats!
As this add-on is required to be released under the GNU General Public License (GPL),
using it will force your libs11n library (and related libs) to fall under the GPL.
To the best of my understanding, this causes your binaries to fall
under the GPL, but not their sources, as the s11n sources do not in any way
depend on MyS, nor do they "even know it exists" (on a technical level, that is).
(If someone would
enlighten me on
the exact license-level implications of this
i would be very appreciative.)
Aside from any potential philosophical or licensing issues, there are
a couple technical ones:
- This code is brand new (only a few hours old, as of the
initial release (15 Aug 2004)), and probably contains some notable bugs. That said,
i have been able to use it with s11n for a wide variety of s11n data and haven't
seen any particular problems yet. Theoretically, saving very complex strings may
cause some Grief, but this isn't proven yet (and should be trivial to fix when
it shows up).
- This code does no error checking for things like a broken db connection.
It does properly recognize when it cannot connect at all, but if a connection
dies during the life of the app Grief will probably result. Since the run-time
of s11nconvert is normally only a few seconds, this has not yet become a
practical concern, but it easily could for longer-running s11n applications.
(To be fixed, maybe. Patches are of course welcomed.)
-
Using it requires a running mysql server and having a valid login for it,
with write access to the database (see below). It is possible to create
a user just for this purpose, and configure the build tree with
--mysql-user=your_s11n_user. That does not create the user or
add any rights for it - it only sets the default user name used
by the Serializer when logging in to the db. There are also the following
related configure options: --mysql-db, --mysql-password,
and --mysql-host, all of which are supported by both the
configure script and s11nconvert.
-
MyS is horribly slow: 5-7 nodes per second is about it's
peak, even when using a local mysql server on a single-user machine.
-
It doesn't yet allow very fine control over the db connections. Patches
would be most appreciated (and feel free to mutilate the db-related interface
as much as you need to, just not the Serializer interface!).
- Minor bug: there appears to be some odd whitespace translation
being done at some undetermined point in the data conversions to/from
the server. Some whitespace (ascii code: oct 040, hex 0x20,
dec 32) gets replaced with a different type of space (ascii code: oct
0137, hex 0x5f, dec 95). It doesn't happen for all spaces, only a
few, so it's really odd. Reproduce by: import a file to a db, then back to a
file, and compare the original file with the routed-through-mysql file.
It happens on all data formats.
Db structure
The default database name for MyS is mys11n, but this can be changed
at build-time by passing --mysql-db to the configure script
or at runtime by passing it to s11nconvert.
Here's what the mys11n db code looks like:
CREATE TABLE nodeinfo ( id int(11) NOT NULL auto_increment, parent_id int(11) NOT NULL default '0', // = parent node's nodeinfo.id, or 0 class varchar(100) NOT NULL default '', // node_traits<>::class_name() name varchar(100) NOT NULL default '', // node_traits<>::name() PRIMARY KEY (id) ) TYPE=MyISAM;
CREATE TABLE properties ( node_id int(11) NOT NULL default '0', // = nodeinfo.id pkey varchar(255) NOT NULL default '', // property key pvalue text // property value ) TYPE=MyISAM; CREATE TABLE pseudofiles ( id int(11) NOT NULL auto_increment, // "inode" number ;) node_id int(11) NOT NULL default '0', // = nodeinfo.id name varchar(255) NOT NULL default '', // "filename" PRIMARY KEY (id) ) TYPE=MyISAM;
|
(Import that into your mysql database. You can grab the
SQL file from the source tree, sql/mys11n.mysql.)
If you know you will only use relatively small data, feel free to change
the varchar and text fields to something smaller, and
change the table TYPEs to whatever you need. MyS is lax about property types:
anything that node_traits<>::get/set() can handle, MyS can handle.
If you are ONLY going to store numbers, for example, you could theoretically
change properties.pvalue to an appropriate numeric type (and get
away with it, i mean).
Nodes are stored in so-called pseudofiles,
with each "file" containing exactly one root node, which itself may
contain any number of nodes. Node "filenames" may be any db-legal
string, up to a maximum of 255 characters (or more, if you modify the
db table to accomodate that). Root node names are case-sensitive and
must be unique - an existing object with a duplicate name will be
deleted before a new one is saved, as will any children and properties
of that object. i.e., it behaves similarly to a filesystem, by
"overwriting" nodes. For portability and ease of db maintenance, this
is not enforced at the db-level, but by the Serializer.
Using MyS with s11nconvert
s11nconvert 0.9.6 adds a couple of features to load arbitrary
DLLs and to force a specific Serializer class to try to read
a given input, as demonstrated in the examples below. Normally
the input Serializer is determined automatically
based on a file's magic cookie, but db records don't have
associated files, and therefor can't have a cookie, can they?
Thus s11nconvert now has the -S option to force a specific
input Serializer (which will obviously fail if you try to use
the wrong Serializer to read a data set).
Examples:
Import a data file into the database:
s11nconvert -dl mysql -f my.s11n -s mysql -o MyDbObject // or, more simply: s11nconvert -f my.s11n -s mysql -o MyDbObject
|
In the previous example we load in a file using -f my.s11n, set the
output format with -s mysql[_serializer], and save it to the "file" named
MyDbObject. The -s classloads the Serializer's DLL, if needed,
but we can also use -dl DLLNAME to explicitely load a DLL, and must do
this when the DLL's name does not match the class name of the Serializer (actually,
it's a tad more complicated than that, but that's not important now...).
Export a database pseudo-file to a data file:
s11nconvert -S mysql -f MyDbObject -s simplexml > my.s11n
| (Remember that MyDbObject is not a real file, but a pseudo-file
in the database.)
Here we loaded the DLL, then used -S mysql to force MyS to be
the input Serializer. The -f option tells
s11nconvert to the load "file" named MyDbObject. We set the
ouput Serializer using -s simplexml, and send it to a file
with either > my.s11n or -o my.s11n.
To copy a mysql-hosted object to another object in the database:
s11nconvert -S mysql -f MyDbObject -s mysql -o "My Copied Object"
|
That forces MyS to be used as the input Serializer (-S),
reads in the MyDbObject using that Serializer, sets the output
Serializer (-s), and saves it to a new "file" called "My Copied Object".
Using s11n 0.9.7+ and mysql_serializer 2004.08.16+, if you try to
output a mysql record to a stream (i.e., by omitting the -o
option) MyS will create a set of proxy data which it can later use to
load the object from it's database:
s11nconvert -f in.paren -s mysql #s11n::io::serializer mysql_serializer <!DOCTYPE SerialTree> <mysql_serializer_proxy_node class="s11n::data_node"> <dbhost>localhost</dbhost> <dbname>mys11n</dbname> <dbpassword></dbpassword> <dbuser>s11n</dbuser> <node_id>6715</node_id> </mysql_serializer_proxy_node>
|
Note that -S is not needed when this option is used, as
the special magic cookie effectively does the same thing, but
does so at a much deeper level in the library than -S
does, and also works when loading streams from arbitrary client code
(as opposed to via s11nconvert).
That proxy code can be fed right back into s11nconvert (or client
code) to load the appropriate object:
s11nconvert -f infilename -s simplexml <!DOCTYPE s11n::simplexml> <somenode s11n_class="NoClass" a="()\b" foo="bar" long="this is a long property"> <fred s11n_class="FredClass" key="value" /> <wilma s11n_class="WilmaClass" the="lovely wife" /> <betty s11n_class="BettyClass" value="the apple of Barney's eye" /> <deep s11n_class="Foo"> <deeper s11n_class="Foo" how_deep="really deep!"> <and_deeper s11n_class="Ouch" /> </deeper> </deep> </somenode>
|
The great part is that this support works at such a level that
arbitrary client code need not know if it's using a database, and
client-side tools (like s11nconvert and
the s11n data browser) can read
and write such data just as easily as they would any other.
Command-line options:
s11nconvert passes on all command-line arguments to DLLs it loads, and MyS supports
several options which you can see by loading the DLL and passing --help
to s11nconvert:
s11nconvert -dl mysql_serializer --help <snip> mysql-db: Sets the s11n/mysql database's name. mysql-host: Sets the mysql db server. mysql-password: Sets the mysql user's password. mysql-user: Sets the mysql db user. mysql-verbose: Toggles on verbose mode for mysql_serializer. <snip>
|
(Tip: any client app which initializes acme::argv_parser::args(int,char**) inherently
passes on all arguments to other objects this way.)
Tips:
If you've read this far, you deserve some bonus tips: the -dl DLLNAME
option has a couple of interesting features:
- It has an evil twin, -DL, which works identically but does not consider
a failed DLL load to be an error (i.e., it ignores such errors). -dl, on the
other hand, will fail if a given DLL cannot be loaded.
- It is optional if a library named SERALIZER.so or SERIALIZER_serializer.so
can be found and opened, where SERIALIZER is the name passed to -S
or -s.
- May be passed base-names if DLLNAME.so is in s11n's classpath
(as described in the library manual).
- May be passed multiple times, unlike the other options. Use it
once for each DLL you want to load, e.g. -dl MyDLL -dl YourDLL.
- s11nconvert does nothing with the DLLs except to open them.
i.e., it does not look for any specific symbols in any DLLs.
In the case of DLLs containing s11n registration code, this simple act is enough
to activate the registrations they contain. You may, if you like, use
this feature for opening non-s11n-related DLLs, but i can personally see
little use in doing so in the context of s11nconvert.
Using MyS in client code
First, see the Big Fat Caveats, above.
To use MyS from s11nlite, you have a couple
approaches:
Simply do the following from somewhere in your app (e.g.,
your main() routine):
s11nlite::serializer_class( "mysql_serializer" );
|
Once you have done this, s11nlite::save() will use that
Serializer. If you're connecting to the db using the build-time
defaults, or using parameters passed in to s11nconvert, then this is
all you need to do to make your database the default i/o repository
used by s11nlite.
Alternately, you can load it dynamically:
// if serialize_class() has been set: s11nlite::serializer_base_type * ser = s11nlite::create_serializer(); // otherwise: s11nlite::serializer_base_type * ser = s11nlite::create_serializer( "mysql_serializer" );
| should do the trick. That will load the DLL, if needed (and if it can
be found). Simply loading the DLL will take care of the classloader
and registration work which is needed to plug the little guy into
the core.
If you need to customize the db connection, look at Database.h, then do:
#include <s11n.net/mysql_serializer/mysql_serializer.hpp> #include <s11n.net/mysql_serializer/libsql++.hpp> // If you want to see any error messages, you must // register an error handler: StderrLog errlog; // Create db object and connect... Database db(host,user,password,db,&errlog); if( db.Connected() ) { delete( s11n::io::mysql_database() ); s11n::io::mysql_database( &db ); } else { // bail out... }
| Note that this requires including GPL'd header files,
which will immediately cast your source code into the GPL!
Note also that the approach shown here is slightly different
from that used by the original mysql driver code,
so be aware of that if you happen to also use that code.
|