2005-03-10 03:35:55 +01:00
|
|
|
DDL enhancements in Firebird v2.
|
|
|
|
--------------------------------
|
2005-02-04 10:58:18 +01:00
|
|
|
|
2005-03-10 03:35:55 +01:00
|
|
|
Author: Claudio Valderrama C. <cvalde at usa.net>
|
|
|
|
|
|
|
|
Other DDL enhancements may have their own README file.
|
|
|
|
|
|
|
|
1) Ability to signal SQL NULL via a NULL pointer.
|
|
|
|
(Claudio Valderrama C.)
|
|
|
|
|
|
|
|
Previous to Firebird v2, UDF authors only could guess they got a null value,
|
|
|
|
but they couldn't be sure, so this let to several problems with UDFs. People
|
|
|
|
ended up assuming that a null string would be passed as an empty string, a null
|
|
|
|
numeric would be the same than zero and a null date would mean the base date used
|
|
|
|
by the engine. Of course, for a numeric value, the author only could assume null
|
|
|
|
if the UDF was done for an environments where it's known that value is not possible
|
|
|
|
normally. But several UDFs (including ib_udf supplied with FB) assumed an empty
|
|
|
|
string most likely would mean a null parameter than a string of length zero. The
|
|
|
|
trick may work with CHAR type (the minimal declared CHAR length is one and would
|
|
|
|
contain a blank character normally, so a binary zero in the first position would
|
|
|
|
signal effectively NULL) but it doesn't work with VARCHAR or CSTRING, where length
|
|
|
|
zero is valid. The other solution was to rely on raw descriptors, but this draws
|
|
|
|
people to an area with a lot of things to check, more than they would want to tackle.
|
|
|
|
The biggest problem is that the engine won't obey the declared type for a parameter;
|
|
|
|
it will simply send whatever data it has for that parameter, so the UDF is left to
|
|
|
|
decide whether to reject or to try to convert the parameter to the expected data
|
|
|
|
type. Since UDFs don't have a formal mechanism to signal errors, the returned value
|
|
|
|
will have to be used as an indicator.
|
|
|
|
But the basic problem was how to keep the simplicity of the typical declarations
|
|
|
|
(no descriptors) while at the same time being able to signal null. The engine
|
|
|
|
normally passed UDF parameters by reference and it means in practical terms to
|
|
|
|
pass a pointer to the data. By simply passing a null pointer we can tell the UDF
|
|
|
|
we have SQL NULL, but since we can't afford to crash an unknown number of different
|
|
|
|
public and private UDFs in use that don't expect NULL, we had to enhance the syntax
|
|
|
|
to be able to request explicitly NULL handling. Therefore, only UDFs that are able
|
|
|
|
to deal with the new scenario can request SQL NULL signaling.
|
|
|
|
To avoid adding more keywords, the NULL keyword is appended to the UDF parameter
|
|
|
|
type and this is all the required change. Example:
|
|
|
|
declare external function sample int null returns int by value...;
|
|
|
|
|
|
|
|
If you are already using functions from ib_udf and want to take advantage of
|
|
|
|
null signaling (and null recognition) in some functions, you should connect to
|
|
|
|
your desired database and run
|
|
|
|
upgrade/v2/ib_udf_upgrade.sql
|
|
|
|
and commit afterwards, preferable when no more users are connected to the database.
|
|
|
|
The code in the listed functions in that script has been modified to recognize
|
|
|
|
null only when NULL is signaled by the engine. Therefore, starting with FB v2,
|
|
|
|
rtrim and ltrim no longer assume that an empty string means a NULL string. If you
|
|
|
|
don't upgrade, the functions won't crash. They simply won't be able to detect NULL.
|
|
|
|
If you never have used ib_udf and want to do that, you should connect to your desired
|
|
|
|
database, run
|
|
|
|
udf/ib_udf2.sql
|
|
|
|
and commit afterwards, preferable when no more users are connected to the database.
|
|
|
|
Note the "2" at the end of the name. The original script for FB v1.5 is still
|
|
|
|
available in the same directory.
|
|
|
|
The directories upgrade" and "udf" are inside the home directory of your FB v2
|
|
|
|
installation.
|
|
|
|
|
|
|
|
|
|
|
|
2) Implemented REVOKE ADMIN OPTION FROM user
|
|
|
|
(Dmitry Yemanov.)
|
|
|
|
|
|
|
|
SYSDBA, the database creator or the owner of an object can grant rights on that
|
|
|
|
object to other users. However, those rights can be made inheritable, too. By using
|
|
|
|
WITH GRANT OPTION, the grantor gives the grantee the right to become a grantor of
|
|
|
|
the same rights in turn. This ability can be removed by the original grantor with
|
|
|
|
REVOKE GRANT OPTION FROM user.
|
|
|
|
|
|
|
|
However, there's a second form that involves roles. Instead of specifying the
|
|
|
|
same rights for many users (soon it becomes a maintenance nightmare) you can
|
|
|
|
create a role, assign rights to that role and then grant it to a group of users.
|
|
|
|
By simply changing the role's rights you affect all those users. By using
|
|
|
|
WITH ADMIN OPTION, the grantor (typically the role creator) gives the grantee the
|
|
|
|
right to become a grantor of the same role in turn. Until FB v2, this ability
|
|
|
|
couldn't be removed unless the original grantor fiddles with system tables directly.
|
|
|
|
Now, the ability to grant the role can be removed by the original grantor with
|
|
|
|
REVOKE ADMIN OPTION FROM user.
|
2005-02-04 10:58:18 +01:00
|
|
|
|
|
|
|
|
2005-04-03 08:59:41 +02:00
|
|
|
3) Blob filter's blob types can be declared by mnemonics for known types.
|
|
|
|
(Alex Peshkov.)
|
|
|
|
|
|
|
|
The original allowed syntax for declaring a blob filter was:
|
|
|
|
declare filter <name> input_type <number> output_type <number>
|
|
|
|
entry_point <function_in_library> module_name <library_name>;
|
|
|
|
|
|
|
|
The alternative new syntax is:
|
|
|
|
declare filter <name> input_type <mnemonic> output_type <mnemonic>
|
|
|
|
entry_point <function_in_library> module_name <library_name>;
|
|
|
|
|
|
|
|
where <mnemonic> refers to a subtype known to the engine. Initially they are
|
|
|
|
binary, text and others mostly of internal usage, but if the user is enough
|
|
|
|
brave, having written a new name in rdb$types, that name could be used, since
|
|
|
|
it's parsed only at declaration time. The engine keeps the numerical value.
|
|
|
|
Remember, only negative subtype values are meant to be defined by users. To get
|
|
|
|
the predefined types, do
|
|
|
|
|
|
|
|
select * from rdb$types where rdb$field_name = 'RDB$FIELD_SUB_TYPE';
|
|
|
|
|
|
|
|
RDB$FIELD_NAME RDB$TYPE RDB$TYPE_NAME RDB$DESCRIPTION RDB$SYSTEM_FLAG
|
|
|
|
=================== ======== ========================== =============== ===============
|
|
|
|
|
|
|
|
RDB$FIELD_SUB_TYPE 0 BINARY <null> 1
|
|
|
|
RDB$FIELD_SUB_TYPE 1 TEXT <null> 1
|
|
|
|
RDB$FIELD_SUB_TYPE 2 BLR <null> 1
|
|
|
|
RDB$FIELD_SUB_TYPE 3 ACL <null> 1
|
|
|
|
RDB$FIELD_SUB_TYPE 4 RANGES <null> 1
|
|
|
|
RDB$FIELD_SUB_TYPE 5 SUMMARY <null> 1
|
|
|
|
RDB$FIELD_SUB_TYPE 6 FORMAT <null> 1
|
|
|
|
RDB$FIELD_SUB_TYPE 7 TRANSACTION_DESCRIPTION <null> 1
|
|
|
|
RDB$FIELD_SUB_TYPE 8 EXTERNAL_FILE_DESCRIPTION <null> 1
|
|
|
|
|
|
|
|
Examples.
|
|
|
|
|
|
|
|
Original declaration:
|
|
|
|
declare filter pesh input_type 0 output_type 3 entry_point 'f' module_name 'p';
|
|
|
|
Alternative declaration:
|
|
|
|
declare filter pesh input_type binary output_type acl entry_point 'f' module_name 'p';
|
|
|
|
|
|
|
|
Bizarre declaration for user defined blob subtype. Remember to commit after the insertion:
|
|
|
|
SQL> insert into rdb$types values('RDB$FIELD_SUB_TYPE', -100, 'XDR', 'test type', 0);
|
|
|
|
SQL> commit;
|
|
|
|
SQL> declare filter pesh2 input_type xdr output_type text entry_point 'p2' module_name 'p';
|
|
|
|
SQL> show filter pesh2;
|
|
|
|
BLOB Filter: PESH2
|
|
|
|
Input subtype: -100 Output subtype: 1
|
|
|
|
Filter library is p
|
|
|
|
Entry point is p2
|
|
|
|
|
2005-05-17 09:21:56 +02:00
|
|
|
|
|
|
|
4) Allow comments in database objects. Proposed syntax for testing:
|
|
|
|
|
|
|
|
COMMENT ON DATABASE IS {'txt'|NULL};
|
|
|
|
COMMENT ON <basic_type> name IS {'txt'|NULL};
|
|
|
|
COMMENT ON COLUMN tblviewname.fieldname IS {'txt'|NULL};
|
|
|
|
COMMENT ON PARAMETER procname.parname IS {'txt'|NULL};
|
|
|
|
|
|
|
|
An empty literal string '' will act as NULL since the internal code (DYN in this case)
|
|
|
|
works this way with blobs.
|
|
|
|
|
|
|
|
basic_type:
|
|
|
|
- DOMAIN
|
|
|
|
- TABLE
|
|
|
|
- VIEW
|
|
|
|
- PROCEDURE
|
|
|
|
- TRIGGER
|
|
|
|
- EXTERNAL FUNCTION
|
|
|
|
- FILTER
|
|
|
|
- EXCEPTION
|
|
|
|
- GENERATOR
|
|
|
|
- SEQUENCE
|
|
|
|
- INDEX
|
|
|
|
- ROLE
|
|
|
|
- CHARACTER SET
|
|
|
|
- COLLATION
|
|
|
|
- SECURITY CLASS (not implemented because Borland hid them).
|
|
|
|
|