Oracle 23c: New features

Oracle 23c: New features

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 way an application consumes and represents data (i. e. JSON documents) and
  • the paradigm in which a relational database stores the data (i. e. rows).

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

  • SQL
  • traditional HTTP /REST operations, i. e. GET, PUT and POST HTTP requests . (see concurrent reading/writing access below)
  • Simple Oracle Document API (SODA)
  • Oracle Database API for MongoDB
  • ORDS

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

  • They're language independent
  • They're optimized by the database

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

  • data constraint
  • display format
  • a collation

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

  • domain_display
  • domain_order
  • domain_name
  • domain_check

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

See also Connor McDonald's tweet and the follow up on YouTube .

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.

Three new PL/SQL packages :

  • Ubiquitous search with dbms_search
  • dbms_hcheck
  • dbms_sql_firewall

Further improvements might (or do?) include:

  • Semi-Join Materialized View Rewrite
  • Improved returning clause for merge and update statements (See this enhancment suggestion and this Oracle community discussion ).
  • Parameter pretty in method json_element_t.stringify
  • alter system kill session … FORCE
  • Aggregation over interval data types
  • Enhancements for blockchain tables
  • SQL <-> PL/SQL transpiler
  • Globally distributed database Raft repliation: a consensus-based commit protocol, enabling declarative replication which does not need Golden Gate or Data Guard.
  • OJM allows HTTP and TCP access while other OS calls are disabled
  • Sagas for microservices / Lock-free reservations
  • alter system kill session … FORCE to terminate a session more immediatly than using alter system kill session … immediate.
  • The data quality operators fuzzy_match() and phonic_encode() (?)
  • Sample schemas for 23c (on Github )
  • Transparent application continuity
  • Asynchronous programming
  • SQL Property Graph Query Language (ISO SQL/PQL standard)
  • default on null for update statements
  • New system privilege table retention.
  • True Cache
  • Real-time SQL Plan Management
  • Read-only Per-PDB Standby
  • In-database SQL firewall
  • OKafka
  • Oracle text indexes with automatic maintenance
  • Transportable binary XML
  • OAuth 2.0 integration
  • On OCI: Oracle Database Zero Data Loss Autonomous Recovery Service (with which Oracle hopes to address the challenges of ransomware, outages and human errors)
  • On OCI: Oracle Full Stack Disaster Recovery Service (Configuration, monitoring and management of the disaster recovery process for the full stack of technologies, including middleware, databases, networks, storage)
  • Support for Unicode Ideographic Variation Sequences (IVS)

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.

要查看或添加评论,请登录

社区洞察

其他会员也浏览了