ORA-01422: exact fetch returns more than requested number of rows ORA-06512: at “SYS.CDBVIEW_INTERN during upgrade

During a database upgrade vom 11.2.0.4 to 18.4.1 i encountered the following problem:

REASON:
      ERRORS FOUND: During Upgrade
         FILENAME: /data/oracle/product/cfgtoollogs/dbua/upgrade2019-02-22_10-18-20AM/DB/catupgrd1R0.log AT LINE NUMBER: 666432
------------------------------------------------------
Identifier CATALOG 19-02-22 11:00:37
SCRIPT    = [/data/oracle/product/18.4/rdbms/admin/cdcore_cols.sql]
ERROR     = [ORA-01422: exact fetch returns more than requested 
             number of rows ORA-06512: at "SYS.CDBVIEW_INTERNAL", line 199
ORA-06512: at "SYS.CDBVIEW_INTERNAL", line 64
ORA-06512: at "SYS.CDBVIEW", line 6
ORA-06512: at line 1
]
STATEMENT = [BEGIN CDBView.create_cdbview(false,'SYS','DBA_TAB_COLS','CDB_TAB_COLS'); END;
]
------------------------------------------------------

Note that the view name (here: DBA_TAB_COLS) can change!

 

The reason for this to fail is in the file $ORACLE_HOME/rdbms/admin/catcdbviews.sql:

 

  -- Check if oldview already exist, qualify with owner# to avoid ORA-1422
  -- error,if two dictionary view with the same name exists across different
  -- schemas like DBA_DV_STATUS which exists in both SYS and DVSYS schemas.

    begin
      execute immediate 'SELECT 1 FROM OBJ$ WHERE NAME = :1' ||
                       ' AND TYPE# in (2, 4) and owner# = :2'
               into cCheck using oldview, ownerId;

 

The developer already identified one issue with his code, but overlooked another: Whenever a database link gets referenced in a piece of pl/sql code (user code or internal code – does not matter), a object with the same name as referenced by the database link is created in OBJ$.

 

SQL> select obj#, name, remoteowner,linkname from obj$ where TYPE# in (2, 4) and name=’DBA_TAB_COLS’;

 

      OBJ# NAME           REMOTEOWNER   LINKNAME
---------- -------------- ------------- --------------------------
  14000219 DBA_TAB_COLS   SYS           STAGING_SYSTEM
      3334 DBA_TAB_COLS

As you can see we have a second object in OBJ$ named DBA_TAB_COLS for the database link. And this causes the upgrade to fail.

 

The fix is relatively simple: Either get a patch from Oracle or fix the code in catcdbviews.sql yourself:

begin
      execute immediate 'SELECT 1 FROM OBJ$ WHERE NAME = :1' ||
          ' AND TYPE# in (2, 4) and remoteowner is null and owner# = :2'
               into cCheck using oldview, ownerId;

After editing the scirpt, install it by executing catcdbview.sql to apply the changes and restart the upgrade. The upgrade will start from the failed step onwards.

Note that the package you edit here is only used during the database upgrade to transfer comment from non-CDB to CDB views. It is deleted after the upgrade and is not needed or used during normal database operation.

This entry was posted in Oracle in general. Bookmark the permalink.

Leave a Reply

Your email address will not be published. Required fields are marked *