2003-12-08 03:19:18 +08:00
|
|
|
/* $OpenLDAP$ */
|
|
|
|
/* This work is part of OpenLDAP Software <http://www.openldap.org/>.
|
2000-03-17 03:08:22 +08:00
|
|
|
*
|
2017-01-04 04:36:47 +08:00
|
|
|
* Copyright 1999-2017 The OpenLDAP Foundation.
|
2003-12-08 03:19:18 +08:00
|
|
|
* Portions Copyright 1999 Dmitry Kovalev.
|
|
|
|
* Portions Copyright 2002 Pierangelo Mararati.
|
2005-01-02 04:38:40 +08:00
|
|
|
* Portions Copyright 2004 Mark Adamson.
|
2003-12-08 03:19:18 +08:00
|
|
|
* All rights reserved.
|
2002-08-23 16:54:08 +08:00
|
|
|
*
|
2003-12-08 03:19:18 +08:00
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted only as authorized by the OpenLDAP
|
|
|
|
* Public License.
|
2002-08-23 16:54:08 +08:00
|
|
|
*
|
2003-12-08 03:19:18 +08:00
|
|
|
* A copy of this license is available in the file LICENSE in the
|
|
|
|
* top-level directory of the distribution or, alternatively, at
|
|
|
|
* <http://www.OpenLDAP.org/license.html>.
|
|
|
|
*/
|
|
|
|
/* ACKNOWLEDGEMENTS:
|
|
|
|
* This work was initially developed by Dmitry Kovalev for inclusion
|
|
|
|
* by OpenLDAP Software. Additional significant contributors include
|
2005-01-03 18:51:59 +08:00
|
|
|
* Pierangelo Masarati and Mark Adamson.
|
2002-08-23 16:54:08 +08:00
|
|
|
*/
|
|
|
|
/*
|
2003-12-08 03:19:18 +08:00
|
|
|
* The following changes have been addressed:
|
2002-08-23 16:54:08 +08:00
|
|
|
*
|
|
|
|
* Enhancements:
|
|
|
|
* - re-styled code for better readability
|
|
|
|
* - upgraded backend API to reflect recent changes
|
|
|
|
* - LDAP schema is checked when loading SQL/LDAP mapping
|
|
|
|
* - AttributeDescription/ObjectClass pointers used for more efficient
|
|
|
|
* mapping lookup
|
|
|
|
* - bervals used where string length is required often
|
|
|
|
* - atomized write operations by committing at the end of each operation
|
|
|
|
* and defaulting connection closure to rollback
|
|
|
|
* - added LDAP access control to write operations
|
|
|
|
* - fully implemented modrdn (with rdn attrs change, deleteoldrdn,
|
|
|
|
* access check, parent/children check and more)
|
|
|
|
* - added parent access control, children control to delete operation
|
|
|
|
* - added structuralObjectClass operational attribute check and
|
|
|
|
* value return on search
|
|
|
|
* - added hasSubordinate operational attribute on demand
|
|
|
|
* - search limits are appropriately enforced
|
|
|
|
* - function backsql_strcat() has been made more efficient
|
|
|
|
* - concat function has been made configurable by means of a pattern
|
|
|
|
* - added config switches:
|
|
|
|
* - fail_if_no_mapping write operations fail if there is no mapping
|
|
|
|
* - has_ldapinfo_dn_ru overrides autodetect
|
|
|
|
* - concat_pattern a string containing two '?' is used
|
|
|
|
* (note that "?||?" should be more portable
|
|
|
|
* than builtin function "CONCAT(?,?)")
|
|
|
|
* - strcast_func cast of string constants in "SELECT DISTINCT
|
|
|
|
* statements (needed by PostgreSQL)
|
|
|
|
* - upper_needs_cast cast the argument of upper when required
|
|
|
|
* (basically when building dn substring queries)
|
2002-08-29 18:55:48 +08:00
|
|
|
* - added noop control
|
|
|
|
* - added values return filter control
|
|
|
|
* - hasSubordinate can be used in search filters (with limitations)
|
|
|
|
* - eliminated oc->name; use oc->oc->soc_cname instead
|
2002-08-23 16:54:08 +08:00
|
|
|
*
|
|
|
|
* Todo:
|
|
|
|
* - add security checks for SQL statements that can be injected (?)
|
|
|
|
* - re-test with previously supported RDBMs
|
|
|
|
* - replace dn_ru and so with normalized dn (no need for upper() and so
|
|
|
|
* in dn match)
|
|
|
|
* - implement a backsql_normalize() function to replace the upper()
|
|
|
|
* conversion routines
|
|
|
|
* - note that subtree deletion, subtree renaming and so could be easily
|
|
|
|
* implemented (rollback and consistency checks are available :)
|
|
|
|
* - implement "lastmod" and other operational stuff (ldap_entries table ?)
|
|
|
|
* - check how to allow multiple operations with one statement, to remove
|
|
|
|
* BACKSQL_REALLOC_STMT from modify.c (a more recent unixODBC lib?)
|
|
|
|
*/
|
2005-01-02 00:21:55 +08:00
|
|
|
/*
|
2005-01-03 18:51:59 +08:00
|
|
|
* Improvements submitted by (ITS#3432)
|
2005-01-02 00:21:55 +08:00
|
|
|
*
|
|
|
|
* 1. id_query.patch applied (with changes)
|
|
|
|
* 2. shortcut.patch applied (reworked)
|
|
|
|
* 3. create_hint.patch applied
|
2005-01-17 07:12:36 +08:00
|
|
|
* 4. count_query.patch applied (reworked)
|
2005-01-02 00:21:55 +08:00
|
|
|
* 5. returncodes.patch applied (with sanity checks)
|
|
|
|
* 6. connpool.patch under evaluation
|
2005-01-17 07:12:36 +08:00
|
|
|
* 7. modoc.patch under evaluation (requires
|
|
|
|
* manageDSAit and "manage"
|
|
|
|
* access privileges)
|
|
|
|
* 8. miscfixes.patch applied (reworked; other
|
|
|
|
* operations need to load the
|
|
|
|
* entire entry for ACL purposes;
|
|
|
|
* see ITS#3480, now fixed)
|
2005-01-02 00:21:55 +08:00
|
|
|
*
|
|
|
|
* original description:
|
|
|
|
|
|
|
|
Changes that were made to the SQL backend.
|
|
|
|
|
|
|
|
The patches were made against 2.2.18 and can be applied individually,
|
|
|
|
but would best be applied in the numerical order of the file names.
|
|
|
|
A synopsis of each patch is given here:
|
|
|
|
|
|
|
|
|
|
|
|
1. Added an option to set SQL query for the "id_query" operation.
|
|
|
|
|
|
|
|
2. Added an option to the SQL backend called "use_subtree_shortcut".
|
|
|
|
When a search is performed, the SQL query includes a WHERE clause
|
|
|
|
which says the DN must be "LIKE %<searchbase>". The LIKE operation
|
|
|
|
can be slow in an RDBM. This shortcut option says that if the
|
|
|
|
searchbase of the LDAP search is the root DN of the SQL backend,
|
|
|
|
and thus all objects will match the LIKE operator, do not include
|
|
|
|
the "LIKE %<searchbase>" clause in the SQL query (it is replaced
|
|
|
|
instead by the always true "1=1" clause to keep the "AND"'s
|
|
|
|
working correctly). This option is off by default, and should be
|
|
|
|
turned on only if all objects to be found in the RDBM are under the
|
|
|
|
same root DN. Multiple backends working within the same RDBM table
|
|
|
|
space would encounter problems. LDAP searches whose searchbase are
|
|
|
|
not at the root DN will bypass this shortcut and employ the LIKE
|
|
|
|
clause.
|
|
|
|
|
|
|
|
3. Added a "create_hint" column to ldap_oc_mappings table. Allows
|
|
|
|
taking the value of an attr named in "create_hint" and passing it to
|
|
|
|
the create_proc procedure. This is necessary for when an objectClass's
|
|
|
|
table is partition indexed by some indexing column and thus the value
|
|
|
|
in that indexing column cannot change after the row is created. The
|
|
|
|
value for the indexed column is passed into the create_proc, which
|
|
|
|
uses it to fill in the indexed column as the new row is created.
|
|
|
|
|
|
|
|
4. When loading the values of an attribute, the count(*) of the number
|
|
|
|
of values is fetched first and memory is allocated for the array of
|
|
|
|
values and normalized values. The old system of loading the values one
|
|
|
|
by one and running realloc() on the array of values and normalized
|
|
|
|
values each time was badly fragmenting memory. The array of values and
|
|
|
|
normalized values would be side by side in memory, and realloc()'ing
|
|
|
|
them over and over would force them to leapfrog each other through all
|
|
|
|
of available memory. Attrs with a large number of values could not be
|
|
|
|
loaded without crashing the slapd daemon.
|
|
|
|
|
|
|
|
5. Added code to interpret the value returned by stored procedures
|
|
|
|
which have expect_return set. Returned value is interpreted as an LDAP
|
|
|
|
return code. This allows the distinction between the SQL failing to
|
|
|
|
execute and the SQL running to completion and returning an error code
|
|
|
|
which can indicate a policy violation.
|
|
|
|
|
|
|
|
6. Added RDBM connection pooling. Once an operation is finished the
|
|
|
|
connection to the RDBM is returned to a pool rather than closing.
|
|
|
|
Allows the next operation to skip the initialization and authentication
|
|
|
|
phases of contacting the RDBM. Also, if licensing with ODBC places
|
|
|
|
a limit on the number of connections, an LDAP thread can block waiting
|
|
|
|
for another thread to finish, so that no LDAP errors are returned
|
|
|
|
for having more LDAP connections than allowed RDBM connections. An
|
|
|
|
RDBM connection which receives an SQL error is marked as "tainted"
|
|
|
|
so that it will be closed rather than returned to the pool.
|
|
|
|
Also, RDBM connections must be bound to a given LDAP connection AND
|
|
|
|
operation number, and NOT just the connection number. Asynchronous
|
|
|
|
LDAP clients can have multiple simultaneous LDAP operations which
|
|
|
|
should not share the same RDBM connection. A given LDAP operation can
|
|
|
|
even make multiple SQL operations (e.g. a BIND operation which
|
|
|
|
requires SASL to perform an LDAP search to convert the SASL ID to an
|
|
|
|
LDAP DN), so each RDBM connection now has a refcount that must reach
|
|
|
|
zero before the connection is returned to the free pool.
|
|
|
|
|
|
|
|
7. Added ability to change the objectClass of an object. Required
|
|
|
|
considerable work to copy all attributes out of old object and into
|
|
|
|
new object. Does a schema check before proceeding. Creates a new
|
|
|
|
object, fills it in, deletes the old object, then changes the
|
|
|
|
oc_map_id and keyval of the entry in the "ldap_entries" table.
|
|
|
|
|
|
|
|
8. Generic fixes. Includes initializing pointers before they
|
|
|
|
get used in error branch cases, pointer checks before dereferencing,
|
|
|
|
resetting a return code to success after a COMPARE op, sealing
|
|
|
|
memory leaks, and in search.c, changing some of the "1=1" tests to
|
|
|
|
"2=2", "3=3", etc so that when reading slapd trace output, the
|
|
|
|
location in the source code where the x=x test was added to the SQL
|
|
|
|
can be easily distinguished.
|
|
|
|
*/
|
2002-08-23 16:54:08 +08:00
|
|
|
|
|
|
|
#ifndef __BACKSQL_H__
|
|
|
|
#define __BACKSQL_H__
|
2000-03-17 03:08:22 +08:00
|
|
|
|
2005-01-03 18:51:59 +08:00
|
|
|
/* former sql-types.h */
|
|
|
|
#include <sql.h>
|
|
|
|
#include <sqlext.h>
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
SWORD ncols;
|
|
|
|
BerVarray col_names;
|
|
|
|
UDWORD *col_prec;
|
2007-03-13 08:34:37 +08:00
|
|
|
SQLSMALLINT *col_type;
|
2005-01-03 18:51:59 +08:00
|
|
|
char **cols;
|
2011-06-14 04:54:56 +08:00
|
|
|
SQLLEN *value_len;
|
2005-01-03 18:51:59 +08:00
|
|
|
} BACKSQL_ROW_NTS;
|
2002-08-14 01:12:27 +08:00
|
|
|
|
2004-01-20 06:13:15 +08:00
|
|
|
/*
|
|
|
|
* Better use the standard length of 8192 (as of slap.h)?
|
2004-10-02 20:05:42 +08:00
|
|
|
*
|
|
|
|
* NOTE: must be consistent with definition in ldap_entries table
|
2004-01-20 06:13:15 +08:00
|
|
|
*/
|
|
|
|
/* #define BACKSQL_MAX_DN_LEN SLAP_LDAPDN_MAXLEN */
|
2002-08-14 01:12:27 +08:00
|
|
|
#define BACKSQL_MAX_DN_LEN 255
|
|
|
|
|
2002-08-23 16:54:08 +08:00
|
|
|
/*
|
|
|
|
* define to enable very extensive trace logging (debug only)
|
|
|
|
*/
|
|
|
|
#undef BACKSQL_TRACE
|
|
|
|
|
2005-01-18 09:10:01 +08:00
|
|
|
/*
|
|
|
|
* define if using MS SQL and workaround needed (see sql-wrap.c)
|
|
|
|
*/
|
|
|
|
#undef BACKSQL_MSSQL_WORKAROUND
|
|
|
|
|
2005-01-17 07:12:36 +08:00
|
|
|
/*
|
|
|
|
* define to enable values counting for attributes
|
|
|
|
*/
|
|
|
|
#define BACKSQL_COUNTQUERY
|
|
|
|
|
|
|
|
/*
|
|
|
|
* define to enable prettification/validation of values
|
|
|
|
*/
|
|
|
|
#define BACKSQL_PRETTY_VALIDATE
|
|
|
|
|
2004-04-08 17:08:28 +08:00
|
|
|
/*
|
|
|
|
* define to enable varchars as unique keys in user tables
|
2004-10-02 20:05:42 +08:00
|
|
|
*
|
|
|
|
* by default integers are used (and recommended)
|
|
|
|
* for performances. Integers are used anyway in back-sql
|
|
|
|
* related tables.
|
2004-04-08 17:08:28 +08:00
|
|
|
*/
|
2004-04-10 18:17:51 +08:00
|
|
|
#undef BACKSQL_ARBITRARY_KEY
|
2004-04-10 17:33:55 +08:00
|
|
|
|
2010-08-13 07:33:54 +08:00
|
|
|
/*
|
|
|
|
* type used for keys
|
|
|
|
*/
|
|
|
|
#if defined(HAVE_LONG_LONG) && defined(SQL_C_UBIGINT) && \
|
|
|
|
( defined(HAVE_STRTOULL) || defined(HAVE_STRTOUQ) )
|
|
|
|
typedef unsigned long long backsql_key_t;
|
|
|
|
#define BACKSQL_C_NUMID SQL_C_UBIGINT
|
|
|
|
#define BACKSQL_IDNUMFMT "%llu"
|
|
|
|
#define BACKSQL_STR2ID lutil_atoullx
|
|
|
|
#else /* ! HAVE_LONG_LONG || ! SQL_C_UBIGINT */
|
|
|
|
typedef unsigned long backsql_key_t;
|
|
|
|
#define BACKSQL_C_NUMID SQL_C_ULONG
|
|
|
|
#define BACKSQL_IDNUMFMT "%lu"
|
|
|
|
#define BACKSQL_STR2ID lutil_atoulx
|
|
|
|
#endif /* ! HAVE_LONG_LONG */
|
|
|
|
|
2004-11-30 06:04:53 +08:00
|
|
|
/*
|
2005-08-11 00:54:45 +08:00
|
|
|
* define to enable support for syncprov overlay
|
2004-11-30 06:04:53 +08:00
|
|
|
*/
|
|
|
|
#define BACKSQL_SYNCPROV
|
|
|
|
|
2004-08-25 18:41:13 +08:00
|
|
|
/*
|
|
|
|
* define to the appropriate aliasing string
|
2004-10-02 20:05:42 +08:00
|
|
|
*
|
|
|
|
* some RDBMSes tolerate (or require) that " AS " is not used
|
|
|
|
* when aliasing tables/columns
|
2004-08-25 18:41:13 +08:00
|
|
|
*/
|
|
|
|
#define BACKSQL_ALIASING "AS "
|
|
|
|
/* #define BACKSQL_ALIASING "" */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* define to the appropriate quoting char
|
2004-10-02 20:05:42 +08:00
|
|
|
*
|
|
|
|
* some RDBMSes tolerate/require that the aliases be enclosed
|
|
|
|
* in quotes. This is especially true for those that do not
|
|
|
|
* allow keywords used as aliases.
|
2004-08-25 18:41:13 +08:00
|
|
|
*/
|
2005-01-22 01:27:27 +08:00
|
|
|
#define BACKSQL_ALIASING_QUOTE ""
|
|
|
|
/* #define BACKSQL_ALIASING_QUOTE "\"" */
|
|
|
|
/* #define BACKSQL_ALIASING_QUOTE "'" */
|
2004-08-25 18:41:13 +08:00
|
|
|
|
2004-04-10 17:33:55 +08:00
|
|
|
/*
|
|
|
|
* API
|
2004-10-02 20:05:42 +08:00
|
|
|
*
|
|
|
|
* a simple mechanism to allow DN mucking between the LDAP
|
|
|
|
* and the stored string representation.
|
2004-04-10 17:33:55 +08:00
|
|
|
*/
|
|
|
|
typedef struct backsql_api {
|
|
|
|
char *ba_name;
|
2005-01-06 00:23:00 +08:00
|
|
|
int (*ba_config)( struct backsql_api *self, int argc, char *argv[] );
|
|
|
|
int (*ba_destroy)( struct backsql_api *self );
|
|
|
|
|
2004-04-10 17:33:55 +08:00
|
|
|
int (*ba_dn2odbc)( Operation *op, SlapReply *rs, struct berval *dn );
|
|
|
|
int (*ba_odbc2dn)( Operation *op, SlapReply *rs, struct berval *dn );
|
2005-01-06 00:23:00 +08:00
|
|
|
|
|
|
|
void *ba_private;
|
|
|
|
struct backsql_api *ba_next;
|
2011-11-05 13:00:44 +08:00
|
|
|
char **ba_argv;
|
|
|
|
int ba_argc;
|
2004-04-10 17:33:55 +08:00
|
|
|
} backsql_api;
|
2004-04-08 17:08:28 +08:00
|
|
|
|
2004-01-20 06:13:15 +08:00
|
|
|
/*
|
|
|
|
* "structural" objectClass mapping structure
|
|
|
|
*/
|
|
|
|
typedef struct backsql_oc_map_rec {
|
|
|
|
/*
|
|
|
|
* Structure of corresponding LDAP objectClass definition
|
|
|
|
*/
|
2005-01-02 00:21:55 +08:00
|
|
|
ObjectClass *bom_oc;
|
2004-01-20 06:13:15 +08:00
|
|
|
#define BACKSQL_OC_NAME(ocmap) ((ocmap)->bom_oc->soc_cname.bv_val)
|
|
|
|
|
2005-01-02 00:21:55 +08:00
|
|
|
struct berval bom_keytbl;
|
|
|
|
struct berval bom_keycol;
|
2004-01-20 06:13:15 +08:00
|
|
|
/* expected to return keyval of newly created entry */
|
2005-01-02 00:21:55 +08:00
|
|
|
char *bom_create_proc;
|
2004-01-20 06:13:15 +08:00
|
|
|
/* in case create_proc does not return the keyval of the newly
|
|
|
|
* created row */
|
2005-01-02 00:21:55 +08:00
|
|
|
char *bom_create_keyval;
|
2004-01-20 06:13:15 +08:00
|
|
|
/* supposed to expect keyval as parameter and delete
|
|
|
|
* all the attributes as well */
|
2005-01-02 00:21:55 +08:00
|
|
|
char *bom_delete_proc;
|
2004-01-20 06:13:15 +08:00
|
|
|
/* flags whether delete_proc is a function (whether back-sql
|
|
|
|
* should bind first parameter as output for return code) */
|
2005-01-02 00:21:55 +08:00
|
|
|
int bom_expect_return;
|
2010-08-13 07:33:54 +08:00
|
|
|
backsql_key_t bom_id;
|
2005-01-02 00:21:55 +08:00
|
|
|
Avlnode *bom_attrs;
|
|
|
|
AttributeDescription *bom_create_hint;
|
2004-01-20 06:13:15 +08:00
|
|
|
} backsql_oc_map_rec;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* attributeType mapping structure
|
|
|
|
*/
|
|
|
|
typedef struct backsql_at_map_rec {
|
|
|
|
/* Description of corresponding LDAP attribute type */
|
|
|
|
AttributeDescription *bam_ad;
|
2007-04-12 09:02:01 +08:00
|
|
|
AttributeDescription *bam_true_ad;
|
2004-01-20 06:13:15 +08:00
|
|
|
/* ObjectClass if bam_ad is objectClass */
|
|
|
|
ObjectClass *bam_oc;
|
|
|
|
|
|
|
|
struct berval bam_from_tbls;
|
|
|
|
struct berval bam_join_where;
|
|
|
|
struct berval bam_sel_expr;
|
|
|
|
|
|
|
|
/* TimesTen, or, if a uppercase function is defined,
|
|
|
|
* an uppercased version of bam_sel_expr */
|
|
|
|
struct berval bam_sel_expr_u;
|
|
|
|
|
|
|
|
/* supposed to expect 2 binded values: entry keyval
|
|
|
|
* and attr. value to add, like "add_name(?,?,?)" */
|
|
|
|
char *bam_add_proc;
|
|
|
|
/* supposed to expect 2 binded values: entry keyval
|
|
|
|
* and attr. value to delete */
|
|
|
|
char *bam_delete_proc;
|
|
|
|
/* for optimization purposes attribute load query
|
|
|
|
* is preconstructed from parts on schemamap load time */
|
|
|
|
char *bam_query;
|
2005-01-17 07:12:36 +08:00
|
|
|
#ifdef BACKSQL_COUNTQUERY
|
|
|
|
char *bam_countquery;
|
|
|
|
#endif /* BACKSQL_COUNTQUERY */
|
2004-01-20 06:13:15 +08:00
|
|
|
/* following flags are bitmasks (first bit used for add_proc,
|
|
|
|
* second - for delete_proc) */
|
|
|
|
/* order of parameters for procedures above;
|
|
|
|
* 1 means "data then keyval", 0 means "keyval then data" */
|
|
|
|
int bam_param_order;
|
|
|
|
/* flags whether one or more of procedures is a function
|
|
|
|
* (whether back-sql should bind first parameter as output
|
|
|
|
* for return code) */
|
|
|
|
int bam_expect_return;
|
|
|
|
|
|
|
|
/* next mapping for attribute */
|
|
|
|
struct backsql_at_map_rec *bam_next;
|
|
|
|
} backsql_at_map_rec;
|
|
|
|
|
|
|
|
#define BACKSQL_AT_MAP_REC_INIT { NULL, NULL, BER_BVC(""), BER_BVC(""), BER_BVNULL, BER_BVNULL, NULL, NULL, NULL, 0, 0, NULL }
|
|
|
|
|
|
|
|
/* define to uppercase filters only if the matching rule requires it
|
|
|
|
* (currently broken) */
|
|
|
|
/* #define BACKSQL_UPPERCASE_FILTER */
|
|
|
|
|
2005-01-14 08:39:24 +08:00
|
|
|
#define BACKSQL_AT_CANUPPERCASE(at) ( !BER_BVISNULL( &(at)->bam_sel_expr_u ) )
|
2004-01-20 06:13:15 +08:00
|
|
|
|
|
|
|
/* defines to support bitmasks above */
|
|
|
|
#define BACKSQL_ADD 0x1
|
|
|
|
#define BACKSQL_DEL 0x2
|
|
|
|
|
2005-01-14 08:39:24 +08:00
|
|
|
#define BACKSQL_IS_ADD(x) ( ( BACKSQL_ADD & (x) ) == BACKSQL_ADD )
|
|
|
|
#define BACKSQL_IS_DEL(x) ( ( BACKSQL_DEL & (x) ) == BACKSQL_DEL )
|
2004-01-20 06:13:15 +08:00
|
|
|
|
|
|
|
#define BACKSQL_NCMP(v1,v2) ber_bvcmp((v1),(v2))
|
|
|
|
|
|
|
|
#define BACKSQL_CONCAT
|
|
|
|
/*
|
|
|
|
* berbuf structure: a berval with a buffer size associated
|
|
|
|
*/
|
|
|
|
typedef struct berbuf {
|
|
|
|
struct berval bb_val;
|
|
|
|
ber_len_t bb_len;
|
|
|
|
} BerBuffer;
|
|
|
|
|
2005-01-14 08:39:24 +08:00
|
|
|
#define BB_NULL { BER_BVNULL, 0 }
|
2004-01-20 06:13:15 +08:00
|
|
|
|
2007-08-21 22:52:43 +08:00
|
|
|
/*
|
|
|
|
* Entry ID structure
|
|
|
|
*/
|
|
|
|
typedef struct backsql_entryID {
|
|
|
|
/* #define BACKSQL_ARBITRARY_KEY to allow a non-numeric key.
|
|
|
|
* It is required by some special applications that use
|
|
|
|
* strings as keys for the main table.
|
|
|
|
* In this case, #define BACKSQL_MAX_KEY_LEN consistently
|
|
|
|
* with the key size definition */
|
|
|
|
#ifdef BACKSQL_ARBITRARY_KEY
|
|
|
|
struct berval eid_id;
|
|
|
|
struct berval eid_keyval;
|
|
|
|
#define BACKSQL_MAX_KEY_LEN 64
|
|
|
|
#else /* ! BACKSQL_ARBITRARY_KEY */
|
|
|
|
/* The original numeric key is maintained as default. */
|
2010-08-13 07:33:54 +08:00
|
|
|
backsql_key_t eid_id;
|
|
|
|
backsql_key_t eid_keyval;
|
2007-08-21 22:52:43 +08:00
|
|
|
#endif /* ! BACKSQL_ARBITRARY_KEY */
|
|
|
|
|
2010-08-13 07:33:54 +08:00
|
|
|
backsql_key_t eid_oc_id;
|
2007-08-21 22:52:43 +08:00
|
|
|
backsql_oc_map_rec *eid_oc;
|
|
|
|
struct berval eid_dn;
|
|
|
|
struct berval eid_ndn;
|
|
|
|
struct backsql_entryID *eid_next;
|
|
|
|
} backsql_entryID;
|
|
|
|
|
2007-11-23 20:47:53 +08:00
|
|
|
#ifdef BACKSQL_ARBITRARY_KEY
|
|
|
|
#define BACKSQL_ENTRYID_INIT { BER_BVNULL, BER_BVNULL, 0, NULL, BER_BVNULL, BER_BVNULL, NULL }
|
|
|
|
#else /* ! BACKSQL_ARBITRARY_KEY */
|
|
|
|
#define BACKSQL_ENTRYID_INIT { 0, 0, 0, NULL, BER_BVNULL, BER_BVNULL, NULL }
|
|
|
|
#endif /* BACKSQL_ARBITRARY_KEY */
|
|
|
|
|
2005-01-16 02:43:34 +08:00
|
|
|
/* the function must collect the entry associated to nbase */
|
|
|
|
#define BACKSQL_ISF_GET_ID 0x1U
|
|
|
|
#define BACKSQL_ISF_GET_ENTRY ( 0x2U | BACKSQL_ISF_GET_ID )
|
2007-08-21 22:52:43 +08:00
|
|
|
#define BACKSQL_ISF_GET_OC ( 0x4U | BACKSQL_ISF_GET_ID )
|
|
|
|
#define BACKSQL_ISF_MATCHED 0x8U
|
2005-01-16 02:43:34 +08:00
|
|
|
#define BACKSQL_IS_GET_ID(f) \
|
|
|
|
( ( (f) & BACKSQL_ISF_GET_ID ) == BACKSQL_ISF_GET_ID )
|
|
|
|
#define BACKSQL_IS_GET_ENTRY(f) \
|
|
|
|
( ( (f) & BACKSQL_ISF_GET_ENTRY ) == BACKSQL_ISF_GET_ENTRY )
|
2007-08-21 22:52:43 +08:00
|
|
|
#define BACKSQL_IS_GET_OC(f) \
|
|
|
|
( ( (f) & BACKSQL_ISF_GET_OC ) == BACKSQL_ISF_GET_OC )
|
2005-01-16 02:43:34 +08:00
|
|
|
#define BACKSQL_IS_MATCHED(f) \
|
|
|
|
( ( (f) & BACKSQL_ISF_MATCHED ) == BACKSQL_ISF_MATCHED )
|
2004-01-20 06:13:15 +08:00
|
|
|
typedef struct backsql_srch_info {
|
|
|
|
Operation *bsi_op;
|
2004-04-10 17:33:55 +08:00
|
|
|
SlapReply *bsi_rs;
|
2004-01-20 06:13:15 +08:00
|
|
|
|
2004-11-29 08:50:55 +08:00
|
|
|
unsigned bsi_flags;
|
|
|
|
#define BSQL_SF_NONE 0x0000U
|
2004-11-30 09:49:06 +08:00
|
|
|
#define BSQL_SF_ALL_USER 0x0001U
|
|
|
|
#define BSQL_SF_ALL_OPER 0x0002U
|
|
|
|
#define BSQL_SF_ALL_ATTRS (BSQL_SF_ALL_USER|BSQL_SF_ALL_OPER)
|
|
|
|
#define BSQL_SF_FILTER_HASSUBORDINATE 0x0010U
|
|
|
|
#define BSQL_SF_FILTER_ENTRYUUID 0x0020U
|
|
|
|
#define BSQL_SF_FILTER_ENTRYCSN 0x0040U
|
2004-11-29 17:36:22 +08:00
|
|
|
#define BSQL_SF_RETURN_ENTRYUUID (BSQL_SF_FILTER_ENTRYUUID << 8)
|
2005-01-14 08:39:24 +08:00
|
|
|
#define BSQL_ISF(bsi, f) ( ( (bsi)->bsi_flags & f ) == f )
|
|
|
|
#define BSQL_ISF_ALL_USER(bsi) BSQL_ISF(bsi, BSQL_SF_ALL_USER)
|
|
|
|
#define BSQL_ISF_ALL_OPER(bsi) BSQL_ISF(bsi, BSQL_SF_ALL_OPER)
|
|
|
|
#define BSQL_ISF_ALL_ATTRS(bsi) BSQL_ISF(bsi, BSQL_SF_ALL_ATTRS)
|
2004-01-20 06:13:15 +08:00
|
|
|
|
2004-10-04 05:08:54 +08:00
|
|
|
struct berval *bsi_base_ndn;
|
2005-01-02 00:21:55 +08:00
|
|
|
int bsi_use_subtree_shortcut;
|
2004-09-29 07:27:39 +08:00
|
|
|
backsql_entryID bsi_base_id;
|
2004-01-20 06:13:15 +08:00
|
|
|
int bsi_scope;
|
2004-10-04 05:08:54 +08:00
|
|
|
/* BACKSQL_SCOPE_BASE_LIKE can be set by API in ors_scope
|
|
|
|
* whenever the search base DN contains chars that cannot
|
|
|
|
* be mapped into the charset used in the RDBMS; so they're
|
|
|
|
* turned into '%' and an approximate ('LIKE') condition
|
|
|
|
* is used */
|
2004-04-10 17:33:55 +08:00
|
|
|
#define BACKSQL_SCOPE_BASE_LIKE ( LDAP_SCOPE_BASE | 0x1000 )
|
2004-01-20 06:13:15 +08:00
|
|
|
Filter *bsi_filter;
|
|
|
|
time_t bsi_stoptime;
|
|
|
|
|
|
|
|
backsql_entryID *bsi_id_list,
|
2004-08-21 17:38:08 +08:00
|
|
|
**bsi_id_listtail,
|
2004-01-20 06:13:15 +08:00
|
|
|
*bsi_c_eid;
|
|
|
|
int bsi_n_candidates;
|
|
|
|
int bsi_status;
|
|
|
|
|
|
|
|
backsql_oc_map_rec *bsi_oc;
|
|
|
|
struct berbuf bsi_sel,
|
|
|
|
bsi_from,
|
|
|
|
bsi_join_where,
|
|
|
|
bsi_flt_where;
|
|
|
|
ObjectClass *bsi_filter_oc;
|
|
|
|
SQLHDBC bsi_dbh;
|
|
|
|
AttributeName *bsi_attrs;
|
|
|
|
|
|
|
|
Entry *bsi_e;
|
|
|
|
} backsql_srch_info;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Backend private data structure
|
|
|
|
*/
|
2005-01-14 08:39:24 +08:00
|
|
|
typedef struct backsql_info {
|
2004-10-02 20:05:42 +08:00
|
|
|
char *sql_dbhost;
|
|
|
|
int sql_dbport;
|
|
|
|
char *sql_dbuser;
|
|
|
|
char *sql_dbpasswd;
|
|
|
|
char *sql_dbname;
|
|
|
|
|
2002-08-14 01:12:27 +08:00
|
|
|
/*
|
|
|
|
* SQL condition for subtree searches differs in syntax:
|
2002-08-23 16:54:08 +08:00
|
|
|
* "LIKE CONCAT('%',?)" or "LIKE '%'+?" or "LIKE '%'||?"
|
2004-10-02 20:05:42 +08:00
|
|
|
* or smtg else
|
2002-08-14 01:12:27 +08:00
|
|
|
*/
|
2004-10-02 20:05:42 +08:00
|
|
|
struct berval sql_subtree_cond;
|
|
|
|
struct berval sql_children_cond;
|
2007-08-20 08:27:47 +08:00
|
|
|
struct berval sql_dn_match_cond;
|
|
|
|
char *sql_oc_query;
|
|
|
|
char *sql_at_query;
|
|
|
|
char *sql_insentry_stmt;
|
|
|
|
char *sql_delentry_stmt;
|
|
|
|
char *sql_renentry_stmt;
|
|
|
|
char *sql_delobjclasses_stmt;
|
2004-10-02 20:05:42 +08:00
|
|
|
char *sql_id_query;
|
|
|
|
char *sql_has_children_query;
|
2007-08-20 08:27:47 +08:00
|
|
|
char *sql_list_children_query;
|
2004-10-02 20:05:42 +08:00
|
|
|
|
|
|
|
MatchingRule *sql_caseIgnoreMatch;
|
|
|
|
MatchingRule *sql_telephoneNumberMatch;
|
|
|
|
|
|
|
|
struct berval sql_upper_func;
|
|
|
|
struct berval sql_upper_func_open;
|
|
|
|
struct berval sql_upper_func_close;
|
|
|
|
struct berval sql_strcast_func;
|
2011-11-05 13:00:44 +08:00
|
|
|
BerVarray sql_concat_func;
|
|
|
|
char *sql_concat_patt;
|
2004-10-02 20:05:42 +08:00
|
|
|
|
2005-06-09 17:43:20 +08:00
|
|
|
struct berval sql_aliasing;
|
|
|
|
struct berval sql_aliasing_quote;
|
|
|
|
struct berval sql_dn_oc_aliasing;
|
|
|
|
|
2005-01-14 08:39:24 +08:00
|
|
|
AttributeName *sql_anlist;
|
|
|
|
|
2004-10-02 20:05:42 +08:00
|
|
|
unsigned int sql_flags;
|
2002-08-23 16:54:08 +08:00
|
|
|
#define BSQLF_SCHEMA_LOADED 0x0001
|
|
|
|
#define BSQLF_UPPER_NEEDS_CAST 0x0002
|
|
|
|
#define BSQLF_CREATE_NEEDS_SELECT 0x0004
|
|
|
|
#define BSQLF_FAIL_IF_NO_MAPPING 0x0008
|
|
|
|
#define BSQLF_HAS_LDAPINFO_DN_RU 0x0010
|
|
|
|
#define BSQLF_DONTCHECK_LDAPINFO_DN_RU 0x0020
|
|
|
|
#define BSQLF_USE_REVERSE_DN 0x0040
|
2004-09-29 07:27:39 +08:00
|
|
|
#define BSQLF_ALLOW_ORPHANS 0x0080
|
2005-01-02 00:21:55 +08:00
|
|
|
#define BSQLF_USE_SUBTREE_SHORTCUT 0x0100
|
2005-01-14 08:39:24 +08:00
|
|
|
#define BSQLF_FETCH_ALL_USERATTRS 0x0200
|
|
|
|
#define BSQLF_FETCH_ALL_OPATTRS 0x0400
|
|
|
|
#define BSQLF_FETCH_ALL_ATTRS (BSQLF_FETCH_ALL_USERATTRS|BSQLF_FETCH_ALL_OPATTRS)
|
2005-05-12 08:03:50 +08:00
|
|
|
#define BSQLF_CHECK_SCHEMA 0x0800
|
2010-08-11 03:55:28 +08:00
|
|
|
#define BSQLF_AUTOCOMMIT_ON 0x1000
|
2005-01-14 08:39:24 +08:00
|
|
|
|
|
|
|
#define BACKSQL_ISF(si, f) \
|
|
|
|
(((si)->sql_flags & f) == f)
|
2002-08-23 16:54:08 +08:00
|
|
|
|
|
|
|
#define BACKSQL_SCHEMA_LOADED(si) \
|
2005-01-14 08:39:24 +08:00
|
|
|
BACKSQL_ISF(si, BSQLF_SCHEMA_LOADED)
|
2002-08-23 16:54:08 +08:00
|
|
|
#define BACKSQL_UPPER_NEEDS_CAST(si) \
|
2005-01-14 08:39:24 +08:00
|
|
|
BACKSQL_ISF(si, BSQLF_UPPER_NEEDS_CAST)
|
2002-08-23 16:54:08 +08:00
|
|
|
#define BACKSQL_CREATE_NEEDS_SELECT(si) \
|
2005-01-14 08:39:24 +08:00
|
|
|
BACKSQL_ISF(si, BSQLF_CREATE_NEEDS_SELECT)
|
2002-08-23 16:54:08 +08:00
|
|
|
#define BACKSQL_FAIL_IF_NO_MAPPING(si) \
|
2005-01-14 08:39:24 +08:00
|
|
|
BACKSQL_ISF(si, BSQLF_FAIL_IF_NO_MAPPING)
|
2002-08-23 16:54:08 +08:00
|
|
|
#define BACKSQL_HAS_LDAPINFO_DN_RU(si) \
|
2005-01-14 08:39:24 +08:00
|
|
|
BACKSQL_ISF(si, BSQLF_HAS_LDAPINFO_DN_RU)
|
2002-08-23 16:54:08 +08:00
|
|
|
#define BACKSQL_DONTCHECK_LDAPINFO_DN_RU(si) \
|
2005-01-14 08:39:24 +08:00
|
|
|
BACKSQL_ISF(si, BSQLF_DONTCHECK_LDAPINFO_DN_RU)
|
2002-08-23 16:54:08 +08:00
|
|
|
#define BACKSQL_USE_REVERSE_DN(si) \
|
2005-01-14 08:39:24 +08:00
|
|
|
BACKSQL_ISF(si, BSQLF_USE_REVERSE_DN)
|
2004-04-08 17:08:28 +08:00
|
|
|
#define BACKSQL_CANUPPERCASE(si) \
|
2004-10-02 20:05:42 +08:00
|
|
|
(!BER_BVISNULL( &(si)->sql_upper_func ))
|
2004-09-29 07:27:39 +08:00
|
|
|
#define BACKSQL_ALLOW_ORPHANS(si) \
|
2005-01-14 08:39:24 +08:00
|
|
|
BACKSQL_ISF(si, BSQLF_ALLOW_ORPHANS)
|
2005-01-02 00:21:55 +08:00
|
|
|
#define BACKSQL_USE_SUBTREE_SHORTCUT(si) \
|
2005-01-14 08:39:24 +08:00
|
|
|
BACKSQL_ISF(si, BSQLF_USE_SUBTREE_SHORTCUT)
|
|
|
|
#define BACKSQL_FETCH_ALL_USERATTRS(si) \
|
|
|
|
BACKSQL_ISF(si, BSQLF_FETCH_ALL_USERATTRS)
|
|
|
|
#define BACKSQL_FETCH_ALL_OPATTRS(si) \
|
|
|
|
BACKSQL_ISF(si, BSQLF_FETCH_ALL_OPATTRS)
|
|
|
|
#define BACKSQL_FETCH_ALL_ATTRS(si) \
|
|
|
|
BACKSQL_ISF(si, BSQLF_FETCH_ALL_ATTRS)
|
2005-05-12 08:03:50 +08:00
|
|
|
#define BACKSQL_CHECK_SCHEMA(si) \
|
|
|
|
BACKSQL_ISF(si, BSQLF_CHECK_SCHEMA)
|
2010-08-11 03:55:28 +08:00
|
|
|
#define BACKSQL_AUTOCOMMIT_ON(si) \
|
|
|
|
BACKSQL_ISF(si, BSQLF_AUTOCOMMIT_ON)
|
2004-10-03 01:33:32 +08:00
|
|
|
|
|
|
|
Entry *sql_baseObject;
|
2011-11-05 13:00:44 +08:00
|
|
|
char *sql_base_ob_file;
|
2004-10-03 01:33:32 +08:00
|
|
|
#ifdef BACKSQL_ARBITRARY_KEY
|
|
|
|
#define BACKSQL_BASEOBJECT_IDSTR "baseObject"
|
|
|
|
#define BACKSQL_BASEOBJECT_KEYVAL BACKSQL_BASEOBJECT_IDSTR
|
2004-10-17 00:16:57 +08:00
|
|
|
#define BACKSQL_IS_BASEOBJECT_ID(id) (bvmatch((id), &backsql_baseObject_bv))
|
2004-10-03 01:33:32 +08:00
|
|
|
#else /* ! BACKSQL_ARBITRARY_KEY */
|
|
|
|
#define BACKSQL_BASEOBJECT_ID 0
|
2005-01-12 08:28:37 +08:00
|
|
|
#define BACKSQL_BASEOBJECT_IDSTR LDAP_XSTRING(BACKSQL_BASEOBJECT_ID)
|
2004-10-03 01:33:32 +08:00
|
|
|
#define BACKSQL_BASEOBJECT_KEYVAL 0
|
|
|
|
#define BACKSQL_IS_BASEOBJECT_ID(id) (*(id) == BACKSQL_BASEOBJECT_ID)
|
|
|
|
#endif /* ! BACKSQL_ARBITRARY_KEY */
|
|
|
|
#define BACKSQL_BASEOBJECT_OC 0
|
2002-08-23 16:54:08 +08:00
|
|
|
|
2004-10-02 20:05:42 +08:00
|
|
|
Avlnode *sql_db_conns;
|
2007-08-20 08:27:47 +08:00
|
|
|
SQLHDBC sql_dbh;
|
|
|
|
ldap_pvt_thread_mutex_t sql_dbconn_mutex;
|
2004-10-02 20:05:42 +08:00
|
|
|
Avlnode *sql_oc_by_oc;
|
|
|
|
Avlnode *sql_oc_by_id;
|
|
|
|
ldap_pvt_thread_mutex_t sql_schema_mutex;
|
|
|
|
SQLHENV sql_db_env;
|
|
|
|
|
|
|
|
backsql_api *sql_api;
|
2002-08-14 01:12:27 +08:00
|
|
|
} backsql_info;
|
|
|
|
|
|
|
|
#define BACKSQL_SUCCESS( rc ) \
|
|
|
|
( (rc) == SQL_SUCCESS || (rc) == SQL_SUCCESS_WITH_INFO )
|
|
|
|
|
2004-04-16 06:29:52 +08:00
|
|
|
#define BACKSQL_AVL_STOP 0
|
|
|
|
#define BACKSQL_AVL_CONTINUE 1
|
|
|
|
|
2005-01-02 00:21:55 +08:00
|
|
|
/* see ldap.h for the meaning of the macros and of the values */
|
|
|
|
#define BACKSQL_LEGAL_ERROR( rc ) \
|
|
|
|
( LDAP_RANGE( (rc), 0x00, 0x0e ) \
|
|
|
|
|| LDAP_ATTR_ERROR( (rc) ) \
|
|
|
|
|| LDAP_NAME_ERROR( (rc) ) \
|
|
|
|
|| LDAP_SECURITY_ERROR( (rc) ) \
|
|
|
|
|| LDAP_SERVICE_ERROR( (rc) ) \
|
|
|
|
|| LDAP_UPDATE_ERROR( (rc) ) )
|
|
|
|
#define BACKSQL_SANITIZE_ERROR( rc ) \
|
|
|
|
( BACKSQL_LEGAL_ERROR( (rc) ) ? (rc) : LDAP_OTHER )
|
2004-01-20 06:13:15 +08:00
|
|
|
|
2007-03-13 08:34:37 +08:00
|
|
|
#define BACKSQL_IS_BINARY(ct) \
|
|
|
|
( (ct) == SQL_BINARY \
|
|
|
|
|| (ct) == SQL_VARBINARY \
|
|
|
|
|| (ct) == SQL_LONGVARBINARY)
|
|
|
|
|
2010-08-08 06:30:53 +08:00
|
|
|
#ifdef BACKSQL_ARBITRARY_KEY
|
|
|
|
#define BACKSQL_IDFMT "%s"
|
|
|
|
#define BACKSQL_IDARG(arg) ((arg).bv_val)
|
|
|
|
#else /* ! BACKSQL_ARBITRARY_KEY */
|
2010-08-13 07:33:54 +08:00
|
|
|
#define BACKSQL_IDFMT BACKSQL_IDNUMFMT
|
2010-08-08 06:30:53 +08:00
|
|
|
#define BACKSQL_IDARG(arg) (arg)
|
|
|
|
#endif /* ! BACKSQL_ARBITRARY_KEY */
|
|
|
|
|
2005-01-02 00:21:55 +08:00
|
|
|
#endif /* __BACKSQL_H__ */
|
2004-01-20 06:13:15 +08:00
|
|
|
|