mirror of
https://github.com/FirebirdSQL/firebird.git
synced 2025-01-22 20:43:02 +01:00
240 lines
9.7 KiB
Plaintext
240 lines
9.7 KiB
Plaintext
DDL enhancements in Firebird v2.
|
|
--------------------------------
|
|
|
|
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.
|
|
|
|
|
|
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
|
|
|
|
|
|
4) Allow comments in database objects.
|
|
(Claudio Valderrama C.)
|
|
|
|
Proposed syntax for testing:
|
|
|
|
COMMENT ON DATABASE IS {'txt'|NULL};
|
|
COMMENT ON <basic_type> name IS {'txt'|NULL};
|
|
COMMENT ON COLUMN table_or_view_name.field_name IS {'txt'|NULL};
|
|
COMMENT ON PARAMETER procedure_name.param_name 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).
|
|
|
|
|
|
5) Allow setting and dropping default values from table fields.
|
|
(Claudio Valderrama C.)
|
|
|
|
Domains allow to change or drop their default. It seems natural that table fields
|
|
can be manipulated the same way without going directly to the system tables. Here's
|
|
the syntax borrowed from the one to alter domains:
|
|
|
|
ALTER TABLE t ALTER [COLUMN] c SET DEFAULT default_value;
|
|
ALTER TABLE t ALTER [COLUMN] c DROP DEFAULT;
|
|
|
|
Notes:
|
|
- Array fields cannot have a default value.
|
|
- If you change the type of a field, the default may remain in place. This is
|
|
because a field can be given the type of a domain with a default but the field
|
|
itself can override such domain. On the other hand, the field can be given a
|
|
type directly in whose case the default belongs logically to the field (albeit
|
|
the information is kept on an implicit domain created behind scenes).
|
|
|
|
|
|
6) New blob filter restriction.
|
|
(Dmitry Yemanov)
|
|
|
|
In FB2, the pair (input subtype, output subtype) must be unique for blob filter
|
|
declarations. This fix stops ambiguity in deciding which blob filter will be
|
|
executed to go from input type X to output type Z. If you have such problem in
|
|
databases created with earlier versions and you have a backup for them that want
|
|
to restore in FB2, expect to see your database restore being rejected by FB2.
|
|
|
|
|
|
7) ALTER computed fields
|
|
(Adriano dos Santos Fernandes)
|
|
|
|
Syntax:
|
|
|
|
alter table ...
|
|
alter <field> [type <data type>] computed by (<expression>);
|
|
|
|
Notes:
|
|
- You cannot alter a non-COMPUTED field to COMPUTED and vice-versa.
|
|
|
|
Example:
|
|
|
|
create table test (
|
|
n integer,
|
|
dn computed by (n * 2)
|
|
);
|
|
|
|
alter table test
|
|
alter dn computed by (n + n);
|
|
|
|
|
|
8) ALTER VIEW
|
|
(Adriano dos Santos Fernandes)
|
|
|
|
Function:
|
|
|
|
Ability to alter a view without the need to recreate (drop and create) the view and all of it's
|
|
dependencies.
|
|
|
|
Syntax:
|
|
|
|
{ create [ or alter ] | alter } view <view_name> [ ( <field list> ) ]
|
|
as <select statement>
|
|
|
|
Example:
|
|
|
|
create table users (
|
|
id integer,
|
|
name varchar(20),
|
|
passwd varchar(20)
|
|
);
|
|
|
|
create view v_users as
|
|
select name from users;
|
|
|
|
alter view v_users (id, name) as
|
|
select id, name from users;
|
|
|