Oracle 23c: New features
Thiago Azadinho - MBA/OCP/OCE/MCSE
DBA / DBRE / DATABASE ADMINISTRATOR / DEVOPS
BOOLEAN data type in SQL
Oracle 23c introduces the boolean datatype for SQL (after it having been in PL/SQL for quite some time).
Some textual values such as true, false, y, yes will be converted to a boolean value, any numerical value that is not 0 will be converted to true, 0 values to false.
create table tq84_bool_test (val varchar2(10), flg boolean);
insert into tq84_bool_test values ('true' , true );
insert into tq84_bool_test values ('false' , false);
insert into tq84_bool_test values ('null' , null );
insert into tq84_bool_test values ('t' ,'t' );
insert into tq84_bool_test values ('yes' ,'yes' );
insert into tq84_bool_test values ('true' ,'true');
insert into tq84_bool_test values ('f' ,'f' );
insert into tq84_bool_test values ('0' , 0 );
select val from tq84_bool_test where flg;
select val from tq84_bool_test where not flg;
The new datatype comes with the function to_boolean(…).
The introduction of the boolean data type is rather unexpected: Some 20 years ago, Tom Kyte asserted that a boolean datatype is not needed
since
Oracle clients earlier than 23c will display true values as 1 and false values as 0:
SQL> select true, false;
--
-- TRUE FALSE
-- ---- -----
-- 1 0
Trying to use a boolean in a release earlier than 23c would result in a ORA-00902: invalid datatype error message.
Selecting an expression without FROM DUAL
The DUAL table is not needed anymore to select one or more expressions:
select
'Hello world' as txt,
7 * 6 as num,
sysdate as now;
JavaScript Stored procedures
Stored procedures can be written in JavaScript :
create mle module tq84_js language javascript as
export function f(p1, p2) { return p1+p2; }
…
create function f (p1 number, p2 number) return number as mle module tq84_js signature f(number, number);
select f(20, 22) /* Ha: no more dual :-) */;
See also MLE modules .
DROP/CREATE TABLE IF [NOT] EXISTS
The if exists and if not exists clauses of the create table and drop table statements are useful to prevent the ORA-00942: table or view does not exist and ORA-00955: name is already used by an existing object errors:
drop table IF EXIST tq84_t1;
create table IF NOT EXIST tq84_t2 (num number, txt varchar2(10));
GROUP BY on expression aliases/position number
group by can use aliases for expressions (such as the result of a PL/SQL function):
select
count(*),
fnc(a, b) xyz
from
t
group by
xyz
having
xyz > 100
;
Prior to Oracle 23, the previous statement would have thrown a ORA-00904: invalid identifier error message.
If the group_by_position_enabled init parameter is set to true, it's also possible to group by a position number In order to group on a position number (rather than an expression alias):
alter session set group_by_position_enabled = true;
select
category_id,
sum(val)
from
t
group by
1;
Updating tables
Tables can be updated with a condition specified in a (?direct?) join : (i.e. with a from clause after the set clause).
udpate dest d
set d.xxx = …
from src s
where d.abc = s.def;
JSON
JSON Schema
A JSON schema allows to validate JSON documents:
create table tq84_json_schema_test (
obj json validate '
{
"num": number,
…
}
'
);
JSON Relational Duality
JSON Relational Duality Views provide a mapping between
The conversion is between these two worlds is triggered in/with GraphQL.
create or replace JSON DUALITY VIEW
from tq84_root_tab -- The name table sitting at the top of the hierarchy
as
tq84_rel_dual_test
@update @insert @delete
{
num : col_num, -- Mapping between JSON property names and table column names
txt : col_txt
children : tq84_child_tab { -- Nesting objects
…
}
}
Accessing duality views
A duality view can be accessed with
Concurrent reading/writing access
REST GET and REST PUT are stateless operations. Thus, a writing operation might overwrite the modification of another writing request (?mid-air collision? problem).
In order to avoid this problem, Oracle has adapted the concept of the ETag response and If-Match request HTTP header (see also the function sys_row_etag).
Advantages over ORM
Some advantages of duality views over Object Relational Mappings (ORMs) are
Describing a duality view's schema
dbms_json_schema.describe('tq84_rel_dual_test')
$dbHost=…
$schemaName=rene
curl -X GET https://$dbHost/ords/$schemaName/metadata-catalog/tq84_rel_dual_test
Links
See also Chris Saxon's tweet .
Table value constructors
Insert multiple records at once (going by the name of ?table value constructor?):
insert into t1 values
(1, 'first record' ),
(2, 'second record'),
(3, 'third record' ),
(4, 'fourth record');
Similarly in a select statement:
select * from (
values
(1, 'first record' ),
(2, 'second record'),
(3, 'third record' ),
(4, 'fourth record')
)
tmp (num, txt);
Schema Level Privileges
Privileges can be granted on entire schemas:
grant
select any table
on
schema sales
to
rene;
SQL domains
SQL domains can optionally specify a
create domain email_addr as varchar2(99);
create domain month_year as varchar2(7)
constraint month_fmt check (regexp_like(month_year, '^\d\d-\d\d\d\d$'))
display 'Year: ' || substr(month_year, 4, 4) || ', Month: ' || substr(month_year, 1, 2)
order substr(month_year, 4, 4) || substr(month_year, 1, 2)
;
create table financial_report (
id integer primary key,
corrections_to email_addr,
rep_period month_year,
pdf blob
);
insert into financial_report values
( 1, '[email protected]', '03-2023', null ),
( 2, '[email protected]', '08-2023', null ),
( 3, '[email protected]', '02-2023', null ),
( 4, '[email protected]', '06-2023', null );
The order clause of the month_year domains influences the sort order of order by so that it is sorted logically rather than by the by the displayed value:
领英推荐
select
rep_period,
corrections_to,
id
from
financial_report
order by
rep_period;
--
-- REP_PER CORRECTIONS_TO ID
-- ------- -------------- ----------
-- 02-2023 [email protected] 3
-- 03-2023 [email protected] 1
-- 06-2023 [email protected] 4
-- 08-2023 [email protected] 2
domain_display(…) applies the display rules when selecting values from a domain:
select
domain_display(rep_period) repper,
corrections_to
from
financial_report
order by
rep_period;
-- REPPER CORRECTIONS_TO
-- --------------------- ---------------------------------------------------------------------------------------------------
-- Year: 2023, Month: 02 [email protected]
-- Year: 2023, Month: 03 [email protected]
-- Year: 2023, Month: 06 [email protected]
-- Year: 2023, Month: 08 [email protected]
Domains are stored in the data dictionary in dba_domains etc.
SQL functions related to domains include
Developer role
A special developer role (named db_developer_role) can be granted to facilitate developers's jobs.
begin
dbms_developer_admin.grant_privs('rene');
end;
/
2023-04-10: Apparently, dbms_developer_admin is not present in (at least the) Oracle 23c Free edition.
SQL> grant db_developer_role to rene identified by renesSecretPassword;
grant succeeded.
SQL> connect rene/renesSecretPassword@ora23c;
connected.
SQL> select * from session_privs order by privilege;
PRIVILEGE
------------------------------
CREATE ANALYTIC VIEW
CREATE ATTRIBUTE DIMENSION
CREATE CUBE
CREATE CUBE BUILD PROCESS
CREATE CUBE DIMENSION
CREATE DIMENSION
CREATE DOMAIN
CREATE HIERARCHY
CREATE JOB
CREATE MATERIALIZED VIEW
CREATE MINING MODEL
CREATE MLE
CREATE PROCEDURE
CREATE SEQUENCE
CREATE SESSION
CREATE SYNONYM
CREATE TABLE
CREATE TRIGGER
CREATE TYPE
CREATE VIEW
DEBUG CONNECT SESSION
EXECUTE DYNAMIC MLE
FORCE TRANSACTION
ON COMMIT REFRESH
24 rows selected.
The db_developer_role is created in $ORACLE_HOME/rdbms/admin/catdevrol.sql (in which script also the corresponding privileges are granted).
Up to 4096 columns per table
Up to 4096 columns per table (requires max_columns to be set to extended and compatible to be set to 23.0.0).
Object and schema annotations
Key value pair annotations for (some) objects provide metadata for data and schema.
create table t_foo ( … )
annotations (
expected_release 'R4.2',
test_coverage 'no'
);
Annotations are supported for tables, views, materialized views, columns, indexes and domains.
Annotations can be queried in the data dictionary:
select * from user_annotations;
select * from user_annotation_values;
select * from user_annotations_usage;
?Asynchronous? transactions (lock-free reservations)
A numeric column that is marked reservable allows multiple transactions to concurrently update the value.
The following example creates such a table and inserts some test data:
create table tq84_async (
id integer primary key,
val number(5,2) RESERVABLE
);
insert into tq84_async values
( 13, 18.22 ),
( 19, 7.83 ),
( 42, 100.00 ),
( 99, 68.54 );
commit;
A session updates a record in this table and then queries the value (which does not seem to change, i.e. is still 100):
SES 1> update tq84_async set val = val + 10 where id = 42;
SES 1> select * from tq84_async where id = 42;
--
-- ID VAL
-- ---------- ----------
-- 42 100
Another session also updates val (while the first session has not commited the update). This session also sees the value of 100:
SES 2> update tq84_async set val = val + 1 where id = 42;
SES 2> select * from tq84_async where id = 42;
--
-- ID VAL
-- ---------- ----------
-- 42 100
The first session then commits its transaction. It now sees its modification:
SES 1> commit;
SES 1> select * from tq84_async where id = 42;
--
-- ID VAL
-- ---------- ----------
-- 42 110
The second session also sees the update of the first session but still not its own update:
SES 2> select * from tq84_async where id = 42;
--
-- ID VAL
-- ---------- ----------
-- 42 110
Only when the 2nd session commits its transaction, its modification becomes visible also:
SES 2> commit;
SES 2> select * from tq84_async where id = 42;
--
-- ID VAL
-- ---------- ----------
-- 42 111
After the commit in the second session, the first session commits also and then selects the updated record:
SES 1> commit;
SES 1> select * from tq84_tab;
ID VAL
--------- ---------
42 111
A table with a reservable column cannot be dropped (error message ORA-55764: Cannot DROP or MOVE tables with reservable columns). Thus, the column must be first set to not reservable:
alter table tq84_async modify val not reservable;
drop table tq84_async;
See also
Improved error messages
A program of Oracle (that is not strictly related to 23c only) tries to improve error messages :
ORA21> select sum(val), gr_1, gr_2 from T group by gr_1;
ORA-00979: not a GROUP BY expression
ORA21> connect …@ORA23
ORA23> select sum(val), gr_1, gr_2 from T group by gr_1;
ORA-00979: "GR_2": must appear in the GROUP BY clause or
be used in an aggregate function
Some of the messages that seem to have changed include
Misc
Here's a python script to download release 23c related Oracle ducumentation PDF files.
Further improvements might (or do?) include:
Docker image
There is a docker image for the free edition of Oracle 23c:
docker pull container-registry.oracle.com/database/free
However, this image is poorly (if at all) documented, thus I had to make a few dabbles in the dark to get it running usefully.
If the container named ora23c was previously started (and stopped), it must be removed:
$ docker rm ora23c
Start a new container:
docker run --name ora23c -d -p 1521:1521 container-registry.oracle.com/database/free
Check the status of the container until it changes from Up … seconds (health: starting) to Up 2 minutes (healthy):
docker ps
When it was running, I changed the password of sys: opened a shell in the container:
$ docker exec ora23c sh -c "echo 'alter user sys identified by IamTheDBA;' | sqlplus -S / as sysdba"
User altered.
It's also possible to open a shell in the container and run SQLPlus interactively:
$ docker exec -it ora23c bash
bash-4.4$ sqlplus / as sysdba
…
SQL>
With this password, I could then connect to the instance from SQL Developer using the service names FREE and FREEPDB1 (see /opt/oracle/product/23c/dbhomeFree/network/admin/tnsnames.ora)
It should be noted that all changes to the database disappear when the docker container is stopped.