前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >【DB笔试面试701】在Oracle中,如何让普通用户可以杀掉自己用户的会话?

【DB笔试面试701】在Oracle中,如何让普通用户可以杀掉自己用户的会话?

作者头像
小麦苗DBA宝典
发布2019-12-17 11:34:12
1.2K0
发布2019-12-17 11:34:12
举报

题目部分

在Oracle中,如何让普通用户可以杀掉自己用户的会话?

答案部分

普通用户想要杀掉会话必须要具有ALTER SYSTEM的权限,但是由于该权限过大,用户可能使用该权限错杀其他用户的会话,所以,有没有其它办法可以实现该功能呢?该类问题也是DBA工作中常遇到的问题,下面作者给出一种解决方案。

首先,可以创建一个查询自己会话信息的视图,将该视图创建公共同义词,然后创建一个存储过程,该存储过程实现杀掉会话的需要,最后将该存储过程的执行权限赋权给PUBLIC即可解决这个问题。

代码实现过程如下所示:

代码语言:javascript
复制
CREATE OR REPLACE VIEW VW_MYOWNERSESSION_LHR AS
SELECT * FROM V$SESSION WHERE USERNAME = USER;
CREATE OR REPLACE PUBLIC SYNONYM SYN_MYOWNERSESSION_LHR FOR SYS.VW_MYOWNERSESSION_LHR;

创建存储过程用于杀掉会话:

代码语言:javascript
复制
CREATE OR REPLACE PROCEDURE PRO_KILL_MYOWN_SESSION_LHR(P_INST    IN NUMBER,
                                                       P_SID     IN NUMBER,
                                                       P_SERIAL# IN NUMBER) IS
  V_IGNORE  PLS_INTEGER;
  V_VERSION VARCHAR2(10);
  V_INST_ID NUMBER;
BEGIN
  SELECT COUNT(*)
    INTO V_IGNORE
    FROM GV$SESSION D
   WHERE USERNAME = USER
     AND SID = P_SID
     AND SERIAL# = P_SERIAL#
     AND D.INST_ID = P_INST;

  SELECT SUBSTR(V.VERSION, 1, INSTR(V.VERSION, '.') - 1), V.INSTANCE_NUMBER
    INTO V_VERSION, V_INST_ID
    FROM V$INSTANCE V;

  IF (V_IGNORE = 1) THEN

    IF (V_VERSION = '10' AND V_INST_ID <> P_INST) THEN
      RAISE_APPLICATION_ERROR(-20001,
                              'Please connect to 【INSTANCE:' || P_INST ||
                              '】,then retry!');
    ELSIF (V_VERSION = '10' AND V_INST_ID = P_INST) THEN
      EXECUTE IMMEDIATE 'ALTER SYSTEM DISCONNECT SESSION ''' || P_SID || ',' ||
                        P_SERIAL# || ''' IMMEDIATE';
    ELSE
      EXECUTE IMMEDIATE 'ALTER SYSTEM DISCONNECT SESSION ''' || P_SID || ',' ||
                        P_SERIAL# || ',@' || P_INST || ''' IMMEDIATE';
    END IF;
  ELSE
    RAISE_APPLICATION_ERROR(-20002,
                            'You do not own session ''' || P_SID || ',' ||
                            P_SERIAL# ||',@' || P_INST || '''');
  END IF;
END PRO_KILL_MYOWN_SESSION_LHR;
/

CREATE OR REPLACE PUBLIC SYNONYM PRO_KILL_SESSION_LHR FOR SYS.PRO_KILL_MYOWN_SESSION_LHR;
GRANT SELECT ON SYN_MYOWNERSESSION_LHR TO PUBLIC;
GRANT EXECUTE ON PRO_KILL_SESSION_LHR TO PUBLIC;

使用方法如下所示:

代码语言:javascript
复制
SELECT USERENV('INSTANCE'),USERENV('SID') FROM DUAL;
SELECT V.INST_ID, SID,SERIAL#,PADDR,STATUS FROM SYN_MYOWNERSESSION_LHR V WHERE SID=1008 AND V.INST_ID=1 ;--假设上一步查询出来的SID为1008,实例号为1
EXEC PRO_KILL_SESSION_LHR(1,1008,35038);--假设上一步查询出来的SERIAL#为35038

使用示例如下所示:

使用SYS用户杀PMON进程的会话:

代码语言:javascript
复制
SYS@lhrdb21> SELECT A.SID,A.SERIAL#,USERENV('INSTANCE'),USERNAME FROM V$SESSION A WHERE A.PROGRAM LIKE '%PMON%';
       SID    SERIAL# USERENV('INSTANCE') USERNAME
---------- ---------- ------------------- ------------------------------
       125          1                   1
SYS@lhrdb21> EXEC PRO_KILL_SESSION_LHR(1,125,1);
BEGIN PRO_KILL_SESSION_LHR(1,125,1); END;
*
ERROR at line 1:
ORA-20002: You do not own session '125,1,@1'
ORA-06512: at "SYS.PRO_KILL_MYOWN_SESSION_LHR", line 36
ORA-06512: at line 1

由于系统进程的用户名为空,所以,避免了误杀系统进程。

使用SYS用户杀普通用户的会话如下所示

代码语言:javascript
复制
SYS@lhrdb21> SELECT A.SID,A.SERIAL#,USERENV('INSTANCE'),USERNAME FROM V$SESSION A WHERE USERNAME='LHR';
       SID    SERIAL# USERENV('INSTANCE') USERNAME
---------- ---------- ------------------- ------------------------------
        79      16453                   1 LHR
SYS@lhrdb21>  EXEC PRO_KILL_SESSION_LHR(1,79,16453);
BEGIN PRO_KILL_SESSION_LHR(1,79,16453); END;
*
ERROR at line 1:
ORA-20002: You do not own session '79,16453,@1'
ORA-06512: at "SYS.PRO_KILL_MYOWN_SESSION_LHR", line 36
ORA-06512: at line 1
SYS@lhrdb21> conn lhr/lhr
Connected.
LHR@lhrdb21> EXEC PRO_KILL_SESSION_LHR(1,79,16453);
PL/SQL procedure successfully completed.

由于79会话属于LHR用户,所以,避免了误杀其它用户的会话,当使用LHR用户的时候,可以正常杀掉会话。

使用LHR用户杀其它用户的会话:

代码语言:javascript
复制
LHR@lhrdb21> SELECT A.SID,A.SERIAL#,USERENV('INSTANCE'),USERNAME FROM V$SESSION A WHERE USERNAME='LHRTEST';
       SID    SERIAL# USERENV('INSTANCE') USERNAME
---------- ---------- ------------------- ------------------------------
       142      12947                   1 LHRTEST
LHR@lhrdb21> EXEC PRO_KILL_SESSION_LHR(1,142,12947);
BEGIN PRO_KILL_SESSION_LHR(1,142,12947); END;
*
ERROR at line 1:
ORA-20002: You do not own session '142,12947,@1'
ORA-06512: at "SYS.PRO_KILL_MYOWN_SESSION_LHR", line 36
ORA-06512: at line 1

普通用户LHR也不能杀掉其它用户LHRTEST的会话。

& 说明:

有关KILL SESSION的更多内容可以参考我的BLOG:http://blog.itpub.net/26736162/viewspace-2121019和http://blog.itpub.net/26736162/viewspace-2121020

本文选自《Oracle程序员面试笔试宝典》,作者:小麦苗

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2019-12-06,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 DB宝 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
对象存储
对象存储(Cloud Object Storage,COS)是由腾讯云推出的无目录层次结构、无数据格式限制,可容纳海量数据且支持 HTTP/HTTPS 协议访问的分布式存储服务。腾讯云 COS 的存储桶空间无容量上限,无需分区管理,适用于 CDN 数据分发、数据万象处理或大数据计算与分析的数据湖等多种场景。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档