使用闪回查询时 BST 到 GMT Oracle 错误 (ORA-08186)

Posted

技术标签:

【中文标题】使用闪回查询时 BST 到 GMT Oracle 错误 (ORA-08186)【英文标题】:BST to GMT Oracle Error (ORA-08186) whilst using a Flashback Query 【发布时间】:2020-02-23 21:37:18 【问题描述】:

希望有人能启发我...

我今天早上到达办公室,发现自动化(通过 OS CRON)SQL*Plus 脚本在 0100BST 开始失败,并在从 BST 切换到 GMT 期间继续这样做直到 0230BST,并出现以下 Oracle 错误

"*ERROR at line 20:
ORA-08186: invalid timestamp specified"

下面的 SQL 已经编辑了一些识别名称,但在语法上与失败的相同。

 SELECT TO_CHAR( MIN(ioh.mdstamp), 'DD-MON-YY hh24:mi:ss')   First_Update
       , TO_CHAR( MAX(ioh.mdstamp), 'DD-MON-YY hh24:mi:ss')  Last_Update
       , ioh.consignment
       , ioh.order       
       , CASE ioh.ord_type
            WHEN 'T' THEN 'STORE MOVE'
            WHEN 'S' THEN 'CUST DEL'
         END                    Movement_Type
       , ioh.status             Order_Status         
       , ioh.site_id            Site
       , COUNT(\*)               Attempts
    FROM table1
         VERSIONS
          BETWEEN TIMESTAMP 
                  from_tz(cast((sysdate) as timestamp),'Europe/London') - INTERVAL '20' MINUTE
              AND MAXVALUE ioh
   WHERE versions_operation = 'I'
     And ioh.client_id      = 'client1'
     AND ioh.status         = 'Cancelled'
     AND ioh.MSTATUS        = 'Pending'     
GROUP BY ioh.consignment
       , ioh.order
       , ioh.ord_type
       , ioh.status         
       , ioh.site_id
ORDER BY ioh.consignment
       , ioh.order_id
       , 2

环境是:

操作系统:RHEL7.2 DB:Oracle Database 12c 企业版版本 12.1.0.2.0 - 64 位 生产 SQL*Plus:版本 12.1.0.2.0

脚本在通过 OS Cron 调用的 BASH 包装脚本中每 15 分钟执行一次。

sqlplus -s $CONNECT_DCS << ! >> $f_SQLResults
        @$SQL_SCRIPTS/$__SCRIPTNAME_NOEXT.sql;  
        exit;
!

另一个脚本捕获了操作系统的 BST 到 DST 的变化:

---------------------------------------
Checking for rejections due to invalid address
Sun 27 Oct **01:47:11 BST** 2019
got count 0
0 - no rejections
---------------------------------------
---------------------------------------
Checking for rejections due to invalid address
Sun 27 Oct **01:02:10 GMT** 2019
got count 0
0 - no rejections
---------------------------------------

故障在 0100h BST 开始发生,并一直持续到如果在 0200BST 没有发生从 BST 更改为 DST 的情况下的 0230h BST。当我确保时区是“欧洲/伦敦”时,我认为数据库会自动处理这个 DST 更改,但我不明白为什么它没有。我确实想到的一个想法是,它可能与使用 MAX SCN 值或 NLS_TERRITORY bwing 设置为“AMERICA”有关,尽管我曾想过在 sql q 中明确设置时区会解决这个问题。

设置了以下数据库配置:

SQL> select dbtimezone from dual;

DBTIME
------
+00:00

1 row selected.

SQL> select sessiontimezone from dual;

SESSIONTIMEZONE
---------------------------------------------------------------------------
Europe/London

1 row selected.

SQL> select SYSTIMESTAMP from dual;

SYSTIMESTAMP
---------------------------------------------------------------------------
28-OCT-19 09.30.36.653265 AM +00:00

1 row selected.

SQL> !date
Mon 28 Oct 09:30:42 GMT 2019

SQL> set lines 200
SQL> l
  1* select * from v$nls_parameters
SQL> /

PARAMETER                                                        VALUE                                                                CON_ID
---------------------------------------------------------------- ---------------------------------------------------------------- ----------
NLS_LANGUAGE                                                     AMERICAN                                                                  0
NLS_TERRITORY                                                    AMERICA                                                                   0
NLS_CURRENCY                                                     $                                                                         0
NLS_ISO_CURRENCY                                                 AMERICA                                                                   0
NLS_NUMERIC_CHARACTERS                                           .,                                                                        0
NLS_CALENDAR                                                     GREGORIAN                                                                 0
NLS_DATE_FORMAT                                                  DD-MON-RR                                                                 0
NLS_DATE_LANGUAGE                                                AMERICAN                                                                  0
NLS_CHARACTERSET                                                 AL32UTF8                                                                  0
NLS_SORT                                                         BINARY                                                                    0
NLS_TIME_FORMAT                                                  HH.MI.SSXFF AM                                                            0

PARAMETER                                                        VALUE                                                                CON_ID
---------------------------------------------------------------- ---------------------------------------------------------------- ----------
NLS_TIMESTAMP_FORMAT                                             DD-MON-RR HH.MI.SSXFF AM                                                  0
NLS_TIME_TZ_FORMAT                                               HH.MI.SSXFF AM TZR                                                        0
NLS_TIMESTAMP_TZ_FORMAT                                          DD-MON-RR HH.MI.SSXFF AM TZR                                              0
NLS_DUAL_CURRENCY                                                $                                                                         0
NLS_NCHAR_CHARACTERSET                                           AL16UTF16                                                                 0
NLS_COMP                                                         BINARY                                                                    0
NLS_LENGTH_SEMANTICS                                             CHAR                                                                      0
NLS_NCHAR_CONV_EXCP                                              FALSE                                                                     0

19 rows selected.

SQL> SELECT TO_CHAR(SYSTIMESTAMP, 'tzr') FROM dual;

TO_CHAR(SYSTIMESTAMP,'TZR')
--------------------------------
+00:00

1 row selected.

SQL> !date +"%Z %z"
GMT +0000

SQL> !timedatectl
      Local time: Mon 2019-10-28 11:57:36 GMT
  Universal time: Mon 2019-10-28 11:57:36 UTC
        RTC time: Mon 2019-10-28 11:57:36
       Time zone: Europe/London (GMT, +0000)
     NTP enabled: yes
NTP synchronized: yes
 RTC in local TZ: no
      DST active: no
 Last DST change: DST ended at
                  Sun 2019-10-27 01:59:59 BST
                  Sun 2019-10-27 01:00:00 GMT
 Next DST change: DST begins (the clock jumps one hour forward) at
                  Sun 2020-03-29 00:59:59 GMT
                  Sun 2020-03-29 02:00:00 BST

SQL>

如果有人能提供帮助,我将不胜感激。

干杯

PS:刚刚注意到我们的 glogin.sql 有以下行:“alter session set time_zone='Europe/London';”

【问题讨论】:

在理想情况下,当有人在这里用[database-administration] 标记问​​题时,问题向导会弹出一个建议,建议最好在DBA.SE site 上提问 我建议宁可BETWEEN TIMESTAMP (SYSTIMESTAMP - INTERVAL '20' MINUTE) AND MAXVALUE,那么你不应该对时区有任何问题。 TIMESTAMP WITH LOCAL TIME ZONE 的比较始终在 UTC 进行。 @APC 感谢您的提醒。在我发布问题后才意识到有一个 dba 站点。然后我稍后添加了标签,但没有显示弹出窗口。我曾考虑过在那里重新发布,但不确定我是否会因为重复条目而违反一些家规。 #新手 @WernfriedDomscheit 谢谢。恐怕我还没有完全掌握这一点。使用我的代码 'from_tz(cast((sysdate) as timestamp),'Europe/London') - INTERVAL '20' MINUTE' on 27/10/2019 at 0100h BST (27/10/2019 0000h UTC)。如果它正在按照您的建议转换为 UTC,它肯定会返回 27/10/2019 2340h UTC?那么为什么它会因“*ERROR at line 20: ORA-08186: invalid timestamp specified”而失败,因为 0100h 是有效的 BST 和 UTC/GMT 时间? 【参考方案1】:

您需要更改 NLS_TIMENLS_TIMESTAMP 参数。可以通过发出以下命令来应用快速解决方法:

SQL> ALTER SESSION SET NLS_TIME_FORMAT         = 'HH24:MI:SSXFF';
SQL> ALTER SESSION SET NLS_TIMESTAMP_FORMAT    = 'DD/MM/RRRR HH24:MI:SSXFF';
SQL> ALTER SESSION SET NLS_TIME_TZ_FORMAT      = 'HH24:MI:SSXFF TZR';
SQL> ALTER SESSION SET NLS_TIMESTAMP_TZ_FORMAT = 'DD/MM/RRRR HH24:MI:SSXFF TZR';

因为您正在寻找HH24 小时格式

例如查询中的TO_CHAR( MIN(ioh.mdstamp), 'DD-MON-YY hh24:mi:ss')

【讨论】:

谢谢。它们不仅会影响返回/显示的数据吗?我的问题似乎在配置/比较阶段 - 我想! 不客气@Mick。您可以在运行脚本之前发出这些命令。是的,这些仅影响当前会话,但应该可以正常工作。如果要全局解析(整个 DB),则需要更改系统命令,这些命令非常危险,应咨询 DBA。 是的,我知道命令谢谢。我不明白的是它们如何影响跨 DST 边界的日期/时间值的比较。我一直认为您列出的那些命令(可能是错误的)纯粹是出于显示/输出目的。根据您的说法,我需要将数据库区域更改为“英国”以获得全球(系统)解决方案 - 对吗? 我的意思是更改它们不仅是为了显示目的,而且还会影响当前会话@Mick 中的操作样式。你尝试过吗 ?顺便说一句,我没有告诉任何关于领土的事情,而是NLS_TIME* 参数。 是的......我知道你没有提到“TERRITORY”,但我知道你提到的那些 NLS_TIME* 参数使用它来导出默认输出 - 从我一直在阅读的关于它们的内容中得出!再次感谢。

以上是关于使用闪回查询时 BST 到 GMT Oracle 错误 (ORA-08186)的主要内容,如果未能解决你的问题,请参考以下文章

Oracle-闪回技术

oracle闪回查询和闪回数据库

达梦闪回功能与oracle对比使用总结

oracle在没有闪回的情况下怎样找回以删了的表

Oracle学习(十三):闪回

Oracle学习(十三):闪回