jueves, 15 de noviembre de 2012

Indices NOSEGMENT o Virtuales

Desde versiones antiguas existe la posibilidad de crear índices tipo NOSEGMENT o virtuales, que permiten "probar" nuevos índices sin afectar al rendimiento/espacio de los sistemas analizados:

CREATE INDEX TABLA_01_I ON TABLA(COLUMNA) NOSEGMENT;

ALTER SESSION SET "_use_nosegment_indexes" = true;

Ahora analizaríamos la consulta :

EXPLAIN PLAN FOR SELECT COUNT(1) FROM TABLA WHERE COLUMNA=...;

Mostrándonos o nó el uso del índice.

Hay que tener cuidado con estos índices, porque:

1) No aparecen en dba_indexes (sí en dba_objects).
2) Oracle puede mostrar que utiliza el índice en las ejecuciones de consultas, (set autotrace on).

miércoles, 14 de noviembre de 2012

Script - Purgado automatizado de Logs 11g

Para el borrado automatizado de logs en instalaciones 11g, se puede/debe hacer uso del comando adrci, que se puede automatizar con scripting:

fichero: purgado.adrci (retención de 2 meses):

SET HOMEPATH diag/rdbms/bbdd/BBDD

PURGE -age 89280

Ejecución del purgado:

adrci script=purge.adrci

martes, 28 de agosto de 2012

Parámetro INITRANS y evento "enq: TX - row lock contention"

El parámetro de configuración INITRANS se gestiona de forma automática desde hace algunas versiones Oracle, pero, en sistemas con concurrencias dispares a lo largo del tiempo quizás conviene
dejarlo en un valor fijo grande para evitar formateos de bloques con valores bajos de ese parámetro y consecuentes problemas de adquisición de bloqueos (enq: TX - row lock contention).

Para analizar uno o varioas bloques:

alter system dump datafile <file_id> block <Block_id>;
alter system dump datafile <file_id> block min <Block_id min> block max <Block id max>;

Buscamos en el udump ... el fichero de traza generado y la cadena "itc", que indicaría el número de ITL reservados realmente para el bloque analizado.

Si vemos que es insuficiente para nuestra tasa de concurrencia, cambiaríamos el INITRANS de los objetos afectados (suponiendo una concurrencia máxima de 80 sesiones):

ALTER [TABLE|INDEX] ... INITRANS 80;


lunes, 6 de agosto de 2012

Script - Migración Export Import mediante tuberías y scp

Script útil para migraciones entre distintas versiones Oracle y/o cargas de datos, basado en export/import tradicional, mediante tuberías tipo pipe y scp.

La ventaja de este script es que no utiliza espacio para los ficheros de export (va todo a través de la tuberías pipe) y que se lanza desde el servidor destino de la copia, sólo requiere acceso "scp" sin password (authorized_keys).


SID=$1
IP=$2
USRSSH=$3
SID2=$4

ssh $USRSSH@$IP ". \$HOME/.profile;. \$HOME/.bash_profile;rm /tmp/tmp_exports_$SID;mknod /tmp/tmp_exports_$SID p;sleep 5;export ORACLE_SID=$SID;exp userid=\'sys/sys AS SYSDBA\' file=/tmp/tmp_exports_$SID rows=y full=y consistent=y log=/tmp/tmp_exports_$SID.log" &
rm /tmp/tmp_exports_$SID2
mknod /tmp/tmp_exports_$SID2 p
ssh $USRSSH@$IP ". \$HOME/.profile;. \$HOME/.bash_profile;cat /tmp/tmp_exports_$SID" > /tmp/tmp_exports_$SID2 &
export ORACLE_SID=$SID2
imp userid=\'sys/sys AS SYSDBA\' file=/tmp/tmp_exports_$SID2 full=y ignore=y

-- Ejemplo:

./export_import_full.sh SID_ORIG 192.168.0.1 oracle SID_DEST

Se irá generando un export full de la instancia SID_ORIG ubicada en el servidor 192.168.0.1, al que se accede con el usuario "oracle" (usuario de instalación oracle con .profile o .bash_profile con las variables de entorno Oracle configuradas).
Este export se irá importando a través del paso por 2 tuberias pipe (ficheros tmp_exports_$SID y tmp_exports_$SID2) sobre la instancia SID_DEST ubicada en el servidor desde el que estemos lanzando el script y al que estamos conectado con el usuario de instalación oracle con las variables de entorno Oracle configuradas.

lunes, 25 de junio de 2012

Script - Generación de fechas mediante función PIPELINED

Función para la generación dinámica de fechas/horas con parámetros: fecha_inicio, fecha_fin, intervalo (en minutos):

CREATE OR REPLACE TYPE "SYSTEM"."DATE_LIST" is table of date;

CREATE OR REPLACE FUNCTION "SYSTEM"."PIPE_DATE_INTERVAL" (p_start date,p_end date, p_inc number)
return date_list pipelined is
v_intervalos number;
begin
v_intervalos:=(p_end-p_start)*24*60/p_inc;
for i in 0 ..v_intervalos-1 loop
pipe row (p_start +p_inc*i/(24*60));
end loop;
return;
end;


-- Ejemplo:

select column_value from table(system.pipe_date_interval(trunc(sysdate), sysdate,30));

Generaría una lista como la siguiente:

COLUMN_VALUE
-------------------
25/06/2012 00:00:00
25/06/2012 00:30:00
25/06/2012 01:00:00
25/06/2012 01:30:00
25/06/2012 02:00:00
25/06/2012 02:30:00
25/06/2012 03:00:00
25/06/2012 03:30:00
25/06/2012 04:00:00
25/06/2012 04:30:00
25/06/2012 05:00:00
25/06/2012 05:30:00
25/06/2012 06:00:00
25/06/2012 06:30:00
25/06/2012 07:00:00
25/06/2012 07:30:00

lunes, 18 de junio de 2012

Script - Traspaso de estadísticas Oracle entre esquemas

Script para el traspaso de estadísticas Oralce entre dos esquemas de distintas bases de datos, se supone existe una tabla intermedia llamadas "CONEXIONES" ubicada en una base de datos central "DB_CENTRAL" donde se almacenan todas los usuarios y passwords de nuestro pool de bases de datos.

El script va pidiendo el esquema de origen y destino, base de datos de origen y destino.

column FECHA new_value FECHA

select
  TO_CHAR(SYSDATE,'YYYYMMDD')   FECHA
from
  DUAL
/

host mkdir &FECHA

WHENEVER SQLERROR CONTINUE
WHENEVER OSERROR CONTINUE

set serveroutput on size 1000000
set termout on
set verify off
set feedback off
set echo off
set heading off
set pagesize 0
set pause off
set wrap on
set line 500
column lanza format a1000


conn usuario/"password"@DB_CENTRAL

define conexion_orig = ""
define conexion_dest = ""
column conexion_orig format a60 new_value Conexion_orig
column conexion_dest format a60 new_value Conexion_dest
COLUMN cadena_conexion format a60
COLUMN comentarios format a50


-- Esquemas a capturar/volcar estadísticas
accept esquema_orig_stats char prompt 'Filtro de esquema ORIGEN a capturar estadísticas: '
accept esquema_dest_stats char prompt 'Filtro de esquema DESTINO a capturar estadísticas: '

-- Mostramos las conexiones origen
accept cadena_orig char prompt 'Filtro de cadena de conexión ORIGEN (usuario SYSTEM): '

select conn_id, trim(usuario)||'/"'||trim(passw)||'"@'||trim(cadena) as cadena_conexion , SUBSTR(COMENTARIOS,1,50) comentarios
from
    CONEXIONES
where
    upper(usuario) like '%'||UPPER('SYSTEM')||'%' and
    upper(cadena) like '%'||UPPER('&cadena_orig')||'%' and
    (UPPER(comentarios) not like 'NO DISPONIBLE%'  AND
    UPPER(comentarios) not like 'ELIMINA%' OR
    COMENTARIOS IS NULL)
order by comentarios, trim(cadena), trim(usuario)
/

accept conn_id_orig number prompt 'Introduzca el id de la conexión ORIGEN: ';

-- Mostramos las conexiones destino
accept cadena_dest char prompt 'Filtro de cadena de conexión DESTINO (usuario SYSTEM): '

select conn_id, trim(usuario)||'/"'||trim(passw)||'"@'||trim(cadena) as cadena_conexion , SUBSTR(COMENTARIOS,1,50) comentarios
from
    CONEXIONES
where
    upper(usuario) like '%'||UPPER('SYSTEM')||'%' and
    upper(cadena) like '%'||UPPER('&cadena_dest')||'%' and
    (UPPER(comentarios) not like 'NO DISPONIBLE%'  AND
    UPPER(comentarios) not like 'ELIMINA%' OR
    COMENTARIOS IS NULL)
order by comentarios, trim(cadena), trim(usuario)
/

accept conn_id_dest number prompt 'Introduzca el id de la conexión DESTINO: ';

spool &FECHA\copy_stats.sql
select 'copy from '||v_from.cadena_conexion||' to '||v_to.cadena_conexion||' append &esquema_dest_stats..STATS_TABLE using SELECT * from &esquema_orig_stats..STATS_TABLE'
from
    (select conn_id, trim(usuario)||'/"'||trim(passw)||'"@'||trim(cadena) as cadena_conexion
    from
        CONEXIONES
    where
        conn_id=&conn_id_orig
    order by conn_id) v_from,
    (select conn_id, trim(usuario)||'/"'||trim(passw)||'"@'||trim(cadena) as cadena_conexion
    from
        CONEXIONES
    where
        conn_id=&conn_id_dest
    order by conn_id) v_to   
/
spool off;


-- Generamos las cadenas de conexión a origen y destino

select trim(usuario)||'/"'||trim(passw)||'"@'||trim(cadena) as conexion_orig
from CONEXIONES
where
    conn_id=decode(nvl(&conn_id_orig,98),0,98,nvl(&conn_id_orig,98))
/
select trim(usuario)||'/"'||trim(passw)||'"@'||trim(cadena) as conexion_dest
from CONEXIONES
where
    conn_id=decode(nvl(&conn_id_dest,98),0,98,nvl(&conn_id_dest,98))
/
-- Conexion a ORIGEN
conn &conexion_orig

-- Borrado, creación y captura de stadisticas
exec DBMS_STATS.DROP_STAT_TABLE('&esquema_orig_stats','STATS_TABLE');
EXEC DBMS_STATS.CREATE_STAT_TABLE('&esquema_orig_stats' , 'STATS_TABLE');
EXEC DBMS_STATS.EXPORT_SCHEMA_STATS('&esquema_orig_stats', 'STATS_TABLE', NULL, '&esquema_orig_stats');


-- Conexion a DESTINO
conn &conexion_dest
-- Borrado, creación de tabla de estadísticas
exec DBMS_STATS.DROP_STAT_TABLE('&esquema_dest_stats','STATS_TABLE');
EXEC DBMS_STATS.CREATE_STAT_TABLE('&esquema_dest_stats' , 'STATS_TABLE');

-- Copia de estadísticas
@&FECHA\copy_stats.sql

-- Depuramos las estadísticas cargadas
DELETE FROM &esquema_dest_stats..STATS_TABLE WHERE (C1,C4,C5) IN
(SELECT A.C1, A.C4, A.C5
FROM
    &esquema_dest_stats..STATS_TABLE A,
    DBA_TAB_COLUMNS B
WHERE
    A.C1=B.TABLE_NAME(+) AND
    A.C4=B.COLUMN_NAME(+) AND
    A.C5=B.OWNER(+) AND
    B.TABLE_NAME IS NULL);
COMMIT;

exec DBMS_STATS.IMPORT_SCHEMA_STATS('&esquema_dest_stats', 'STATS_TABLE', NULL, '&esquema_dest_stats');


martes, 5 de junio de 2012

Script - Control de Sesiones en base a Planes de Ejecución

Script para el control de sesiones activas en base a métodos de acceso en sus planes de ejecución.

En el ejemplo: Sesiones activas por más de 1 hora con MERGE JOIN CARTESIAN, el control en este caso consiste en cerrar la sesión mediante un kill immediate.

declare
cursor sql_id is
select l.sql_id from V$session l, v$sql s where SUBSTR(l.STATUS,1,10) LIKE 'ACTIV%' and
            l.sql_id is not null and
            l.username is not null and
            l.sql_id=s.sql_id and
            -- CPU_TIME > 1 hora
            s.cpu_time > (3600)*1000000 ;
v_ind number;
v_mata varchar2(1000);
begin
v_ind:=0;
for v_reg in sql_id
loop
    selecT COUNT(1) INTO v_ind
    FROM TABLE(DBMS_XPLAN.DISPLAY_cursor(v_reg.sql_id,null)) where plan_table_output like '%MERGE JOIN CARTESIAN%';
    if v_ind > 0 then
        select 'ALTER SYSTEM KILL SESSION '''||    L.SID||','||SUBSTR(TO_CHAR(L.SERIAL#),1,8)||''' immediate' into v_mata FROM V$SESSION L WHERE  L.sql_id=v_reg.sql_id and l.username is not null;
        execute immediate v_mata;
    end if;
    v_ind:=0;
end loop;
exception
    when others then null;
end;
/

lunes, 4 de junio de 2012

Script - Reescritura de Sentencias

Script para reescribir sentencias mediante el uso del paquete DBMS_ADVANCED_REWRITE.

exec sys.dbms_advanced_rewrite.drop_rewrite_equivalence('ejemplo');
exec sys.dbms_advanced_rewrite.declare_rewrite_equivalence ( -
name => 'ejemplo', -
source_stmt=> q'[select columna1, columna2 from tablaA where columna1 like 'ABCD%' ]', -
destination_stmt=> q'[select columna1, columna2 from tablaB where columna3 = 'ABCD' ]', -
validate       => false, -
rewrite_mode=>'TEXT_MATCH');

Vista donde podemos ver la equivalencia creada: DBA_REWRITE_EQUIVALENCES

ALTER SESSION SET QUERY_REWRITE_ENABLED=FORCE;
ALTER SESSION SET QUERY_REWRITE_INTEGRITY=TRUSTED;

Así cuando ejecutemos :

select columna1, columna2 from tablaA where columna1 like 'ABCD%'

Realmente se ejecutará:

select columna1, columna2 from tablaB where columna3 = 'ABCD'

Útil para situaciones en las que no se puede modificar el código de la aplicación, y el resto de opciones no fuciona (SQL_profiles, hints, parámetros, etc).

IMPORTANTE: No admite el uso de variables BIND, por lo que su ámbito de actuación está limitado a sentencias que no hacen uso de variables bind.


lunes, 14 de mayo de 2012

Scrip - Analiza_AWR

Script para generar estadísticas de sentencias (conociendo su sql_id o alguna parte del texto de la sql).

Genera los planes almacenados para la sentencia elegida y propuestas ADDM.

Tiene como parámetros el txt o sql_id de la consulta, usuario de parseo, snap_id inicial si se quiere indicar, snap_id final si se quiere indicar y fechas fec_ini y fec_fin de lanzamiento de la sentencia, además solicita el min_snap y max_snap para sentencias que han tenido varios planes a lo largo de distintos awr.

Cliente de lanzamiento : sqlplus.

set linesize 4500

define v_sql =""
column v_sql format a400 new_value V_sql
column fecha_Max_plan format a20
column min_snap format 999999
column max_snap format 999999
PROMPT
PROMPT =================================================================
PROMPT
PROMPT LISTA DE CONSULTAS Y SQL_IDS
PROMPT
PROMPT =================================================================
PROMPT
accept sqltxt char prompt 'Introduzca una cadena SQL a buscar o SQL_ID: '
PROMPT
accept snap_id_ini char DEFAULT 0 prompt 'Introduzca el SNAP_ID Inicial de localización de las sentencias: '
PROMPT
accept snap_id_fin char DEFAULT 999999999 prompt 'Introduzca el SNAP_ID Final de localización de las sentencias: '
PROMPT
accept fec_ini char DEFAULT sysdate-4000 prompt 'Introduzca la Fecha Inicial de localización de las sentencias (formato "sysdate - ..."): '
PROMPT
accept fec_fin char DEFAULT sysdate prompt 'Introduzca la Fecha Inicial de localización de las sentencias (formato "sysdate - ..."): '
PROMPT

select plan_hash_value,to_date(MAX(cast(END_INTERVAL_TIME as DATE)),'dd/mm/yyyy hh24:mi:ss') Fecha_Max_plan,
min(q.snap_id) Min_Snap, max(q.snap_id) Max_Snap,
T.SQL_ID, SUM(EXECUTIONS_DELTA) EJECUCIONES, ROUND(SUM(ELAPSED_TIME_DELTA)/1000000,2) "Elapsed Time(s)", SUM(DISK_READS_DELTA) "Physical Reads",
ROUND(SUM(ELAPSED_TIME_DELTA)/(1000000*SUM(EXECUTIONS_DELTA)),2) "Elap per Exec(s)/EJEC",
ROUND(SUM(DISK_READS_DELTA)/SUM(EXECUTIONS_DELTA),2) "Reads per Exec",
--replace(TO_CHAR(SUBSTR(T.SQL_TEXT,1,4000)),chr(39),chr(39)||'||chr(39)||'||chr(39)) v_sql
TO_CHAR(SUBSTR(T.SQL_TEXT,1,4000)) v_sql
        from
            dba_hist_sqlstat q,
            dba_hist_snapshot s,
            (select SQL_ID, SUBSTR(SQL_TEXT,1,4000) SQL_TEXT from dba_hist_sqltext where UPPER(sql_text) like '%'||UPPER('&sqltxt')||'%' or LOWER(sql_id)=LOWER('&sqltxt')) T
        where q.sql_id = T.SQL_ID
            and q.snap_id = s.snap_id
            and q.snap_id BETWEEN (case
                                                when &snap_id_ini=&snap_id_fin then &snap_id_ini
                                                else &snap_id_ini +1 end) AND &snap_id_fin
            and s.begin_interval_time between &fec_ini and &fec_fin
            and executions_delta > 0
GROUP BY PLAN_HASH_VALUE, T.SQL_ID,
--replace(TO_CHAR(SUBSTR(T.SQL_TEXT,1,4000)),chr(39),chr(39)||'||chr(39)||'||chr(39))
TO_CHAR(SUBSTR(T.SQL_TEXT,1,4000))
/

accept sqlid char prompt 'Introduzca el SQL_ID: '
accept min_snap char prompt 'Introduzca el Min_Snap: '
accept max_snap char prompt 'Introduzca el Max_Snap: '

PROMPT
PROMPT =================================================================
PROMPT
PROMPT CONSULTA ELEGIDA PARA ANALISIS
PROMPT
PROMPT =================================================================
PROMPT

SELECT replace(TO_CHAR(SUBSTR(T.SQL_TEXT,1,4000)),chr(39),chr(39)||'||chr(39)||'||chr(39)) v_sql
FROM dba_hist_sqltext T WHERE SQL_ID='&sqlid'
/
   

PROMPT
PROMPT =================================================================
PROMPT
PROMPT PLANES ALMACENADOS EN AWR
PROMPT
PROMPT =================================================================
PROMPT

SELECT * FROM TABLE(DBMS_XPLAN.DISPLAY_AWR('&sqlid'));

   
exec dbms_sqltune.drop_tuning_task('sql_tuning_task');
DECLARE
  my_sqltext CLOB;
  task_name VARCHAR2(30);
BEGIN
  my_sqltext := '&v_sql';
  task_name := DBMS_SQLTUNE.CREATE_TUNING_TASK(
                                                                   begin_snap=> &min_snap,
                                                                   end_snap=> &max_snap,
                                                                   sql_id => '&sqlid',
                                   --bind_list => sql_binds(anydata.Convertvarchar2('noticia'),anydata.Convertvarchar2('noticia'),anydata.Convertvarchar2('10')),
                                   --user_name => '&usuario',
                                   scope => 'COMPREHENSIVE',
                                   time_limit => 60,
                                   task_name => 'sql_tuning_task');
END;
/
exec dbms_sqltune.execute_tuning_task ( 'sql_tuning_task');

PROMPT
PROMPT =================================================================
PROMPT
PROMPT PLAN DE EJECUCION ACTUAL Y PROPUESTAS
PROMPT
PROMPT =================================================================
PROMPT
COLUMN RESULTADO FORMAT A1000
select dbms_sqltune.report_tuning_task('sql_tuning_task')  AS RESULTADO from dual;

Script - Analiza_SGA

Script para generar estadísticas de sentencias (conociendo su sql_id o alguna parte del texto de la sql), utilizando como variables bind las máximas utilizadas en el último snap_id almacenado.

Genera los planes almacenados para la sentencia elegida y posteriormente muestra el análisis actual (con la sustitución de las variables bind) y propuestas ADDM.

Tiene como parámetros el txt o sql_id de la consulta, usuario de parseo, snap_id inicial si se quiere indicar, snap_id final si se quiere indicar y fechas fec_ini y fec_fin de lanzamiento de la sentencia:

Cliente de lanzamiento : sqlplus.


set linesize 4500

define v_sql =""
column v_sql format a4000 new_value V_sql
column fecha_Max_plan format a20
PROMPT
PROMPT =================================================================
PROMPT
PROMPT LISTA DE CONSULTAS Y SQL_IDS
PROMPT
PROMPT =================================================================
PROMPT
accept sqltxt char prompt 'Introduzca una cadena SQL a buscar o SQL_ID (en AWR y SGA): '
PROMPT
accept usuario char DEFAULT SYSTEM prompt 'Introduzca el usuario de Parseo de la sentencia: '
PROMPT
accept snap_id_ini char DEFAULT 0 prompt 'Introduzca el SNAP_ID Inicial de localización de las sentencias: '
PROMPT
accept snap_id_fin char DEFAULT 999999999 prompt 'Introduzca el SNAP_ID Final de localización de las sentencias: '
PROMPT
accept fec_ini char DEFAULT sysdate-4000 prompt 'Introduzca la Fecha Inicial de localización de las sentencias (formato "sysdate - ..."): '
PROMPT
accept fec_fin char DEFAULT sysdate prompt 'Introduzca la Fecha Inicial de localización de las sentencias (formato "sysdate - ..."): '
PROMPT

select plan_hash_value,to_date(MAX(cast(END_INTERVAL_TIME as DATE)),'dd/mm/yyyy hh24:mi:ss') Fecha_Max_plan, T.SQL_ID, SUM(EXECUTIONS_DELTA) EJECUCIONES, ROUND(SUM(ELAPSED_TIME_DELTA)/1000000,2) "Elapsed Time(s)", SUM(DISK_READS_DELTA) "Physical Reads",
ROUND(SUM(ELAPSED_TIME_DELTA)/(1000000*SUM(EXECUTIONS_DELTA)),2) "Elap per Exec(s)/EJEC",
ROUND(SUM(DISK_READS_DELTA)/SUM(EXECUTIONS_DELTA),2) "Reads per Exec",
replace(TO_CHAR(SUBSTR(T.SQL_TEXT,1,4000)),chr(39),chr(39)||'||chr(39)||'||chr(39)) v_sql
        from
            dba_hist_sqlstat q,
            dba_hist_snapshot s,
            (select SQL_ID, SUBSTR(SQL_TEXT,1,4000) SQL_TEXT from dba_hist_sqltext where UPPER(sql_text) like '%'||UPPER('&sqltxt')||'%' or LOWER(sql_id)=LOWER('&sqltxt')) T
        where q.sql_id = T.SQL_ID
            and q.snap_id = s.snap_id
            and q.snap_id BETWEEN (case
                                                when &snap_id_ini=&snap_id_fin then &snap_id_ini
                                                else &snap_id_ini +1 end) AND &snap_id_fin
            and s.begin_interval_time between &fec_ini and &fec_fin
            and executions_delta > 0
GROUP BY PLAN_HASH_VALUE, T.SQL_ID,
replace(TO_CHAR(SUBSTR(T.SQL_TEXT,1,4000)),chr(39),chr(39)||'||chr(39)||'||chr(39))
union
select plan_hash_value, TO_DATE(to_date(LAST_LOAD_TIME,'YYYY-MM-DD/hh24:mi:ss'),'dd/mm/yyyy hh24:mi:ss') Fecha_Max_plan, T.SQL_ID, sum(executions) ejecucions, ROUND(SUM(ELAPSED_TIME)/1000000,2) "Elapsed Time(s)", SUM(DISK_READS) "Physical Reads",
    ROUND(SUM(ELAPSED_TIME)/(1000000*SUM(EXECUTIONS)),2) "Elap per Exec(s)/EJEC",
    ROUND(SUM(DISK_READS)/SUM(EXECUTIONS),2) "Reads per Exec",
    replace(TO_CHAR(SUBSTR(T.SQL_fullTEXT,1,4000)),chr(39),chr(39)||'||chr(39)||'||chr(39)) v_sql
from
    v$sql T
where
    (UPPER(sql_text) like '%'||UPPER('&sqltxt')||'%' or LOWER(sql_id)=LOWER('&sqltxt') )
    AND EXECUTIONS > 0
GROUP BY PLAN_HASH_VALUE, T.SQL_ID, TO_DATE(to_date(LAST_LOAD_TIME,'YYYY-MM-DD/hh24:mi:ss'),'dd/mm/yyyy hh24:mi:ss'),
replace(TO_CHAR(SUBSTR(T.SQL_FULLTEXT,1,4000)),chr(39),chr(39)||'||chr(39)||'||chr(39))    
/

accept sqlid char prompt 'Introduzca el SQL_ID: '

PROMPT
PROMPT =================================================================
PROMPT
PROMPT CONSULTA ELEGIDA PARA ANALISIS
PROMPT
PROMPT =================================================================
PROMPT

SELECT replace(TO_CHAR(SUBSTR(T.SQL_TEXT,1,4000)),chr(39),chr(39)||'||chr(39)||'||chr(39)) v_sql
FROM dba_hist_sqltext T WHERE SQL_ID='&sqlid'
/
   

PROMPT
PROMPT =================================================================
PROMPT
PROMPT PLANES ALMACENADOS EN AWR
PROMPT
PROMPT =================================================================
PROMPT

SELECT * FROM TABLE(DBMS_XPLAN.DISPLAY_AWR('&sqlid'));

-- AHORA NOS QUEDAMOS CON LOS ÚLTIMOS VALORES INFORMADOS SEGÚN DBA_HIST_SQLBIND
var sqlidd varchar2(4000);
begin
:sqlidd:='&v_sql';
FOR rec in (select name, value_string from DBA_HIST_SQLBIND WHERE SQL_ID='&sqlid' and snap_id =(select max(snap_id) from DBA_HIST_SQLBIND WHERE SQL_ID='&sqlid'))
LOOP
    :sqlidd := substr(replace('&v_sql',rec.name,CHR(39)||rec.value_string||CHR(39)),1,4000);
END LOOP;
end;
/
column sqlid2 format a4000 new_value Sqlid
select replace(:sqlidd,chr(39),chr(39)||'||chr(39)||'||chr(39)) v_sql from dual;

exec dbms_sqltune.drop_tuning_task('sql_tuning_task');
DECLARE
  my_sqltext CLOB;
  task_name VARCHAR2(30);
BEGIN
  my_sqltext := '&v_sql';
  task_name := DBMS_SQLTUNE.CREATE_TUNING_TASK( sql_text => my_sqltext,
                                   --bind_list => sql_binds(anydata.Convertvarchar2('noticia'),anydata.Convertvarchar2('noticia'),anydata.Convertvarchar2('10')),
                                   user_name => '&usuario',
                                   scope => 'COMPREHENSIVE',
                                   time_limit => 60,
                                   task_name => 'sql_tuning_task');
END;
/
exec dbms_sqltune.execute_tuning_task ( 'sql_tuning_task');

PROMPT
PROMPT =========================================================================
PROMPT
PROMPT PLAN DE EJECUCION ACTUAL(CON SUSTITUCIÓN DE VARIABLES BIND) Y PROPUESTAS
PROMPT
PROMPT =========================================================================
PROMPT
COLUMN RESULTADO FORMAT A1000
select dbms_sqltune.report_tuning_task('sql_tuning_task')  AS RESULTADO from dual;



martes, 24 de abril de 2012

Script - Restore Datapump por Red

Procedimiento para exportación/importación mediante Datapump via dblink, que debe ser creado en la base de datos Destino:

CREATE OR REPLACE PROCEDURE RECUPERA
         ( source_schema in varchar2,
           destination_schema in varchar2,
           network_link in varchar2 default 'DBLINK')
as
  JobHandle   number;
  js  varchar2(9);
  q   varchar2(1) := chr(39);
BEGIN 
   JobHandle := dbms_datapump.open ('IMPORT','SCHEMA',network_link);
  dbms_datapump.metadata_filter ( JobHandle,'SCHEMA_LIST',q||source_schema||q);
    dbms_datapump.metadata_remap ( JobHandle,'REMAP_SCHEMA',source_schema,destination_schema);
   dbms_datapump.set_parameter ( JobHandle,'TABLE_EXISTS_ACTION','REPLACE' );
  dbms_datapump.start_job( JobHandle);  
  dbms_datapump.wait_for_job( JobHandle, js);  
end;
/

Ejecución:

exec recupera('ESQUEMA_ORIGEN', 'ESQUEMA_DESTINO', 'DB_LINK_A_ORIGEN');

domingo, 5 de febrero de 2012

Banco de Datos Cloud (II)

Apoyándonos en la tecnología Grid de Oracle hemos diseñado e implantado un Banco de Datos heterogéneo, que da servicio de acceso a bases de datos Oracle en RAC y MySQL en Alta Disponibilidad CLUSTER.

MySQL Cluster (motor de almacenamiento NDB) se basa en el concepto de "Shared Nothing", que es justo lo contrario de la tecnología de clusterización Oracle.

La solución implantada de gestión de un Cluster MySQL bajo la protección Grid de Oracle no desvirtúa nada de ambos conceptos, para MySQL en Cluster todo sigue funcionando en base a servidores de Almacenamiento NDB, servidor de administración MGF y Apis MYSQLD, todo distribuido por ips/servidores independientes.  La capa Grid de Oracle sustenta la gestión de las ips/almacenamiento y disponibilidad de los servicios MySQL en Cluster de una forma sencilla, controlable y centralizada.

Las ventajas de este diseño incrustado de MySQL Cluster sobre Grid Oracle responden a las siguientes necesidades:

¿Cómo aumento el rendimiento de mi Entrada/Salida en mi Cluster MySQL? => Incluímos más discos/luns en la capa ASM de Oracle que provisiona el espacio ACFS para MySQL.

¿Cómo aumento la capacidad de Procesamiento de mi Cluster MySQL? => Incluímos más nodos en la infraestructura Grid.

Para más información, contacte con nosotros.

sábado, 4 de febrero de 2012

Estilos Word (copiar de un documento a otro)

Para copiar estilos de un documento Word a otro:

http://office.microsoft.com/es-es/word-help/copiar-estilos-en-otro-documento-o-plantilla-HP005189264.aspx


  1. En el menú Herramientas, haga clic en Plantillas y complementos.
  2. Haga clic en Organizador y, a continuación, en la ficha Estilos.
  3. Para copiar elementos a o desde otra plantilla u otro archivo, haga clic en Cerrar archivo para cerrar el documento activo y su plantilla adjunta, o bien para cerrar la plantilla Normal. A continuación, haga clic en Abrir archivo y abra la plantilla o el archivo que desee.
  4. Haga clic en los elementos que desee copiar de cualquiera de las dos listas y, a continuación, en Copiar.


lunes, 9 de enero de 2012

Instalación - Oracle Enterprise Manager 12c

Instalación Oracle Enterprise Manager 12c
 
Instalaremos OEM 12C sobre un nodo que previamente había sido parte de un cluster oracle 10g, intentaremos que funcione sobre un portátil (3Gb, 2 cores) virtualizado con virtualbox.

Reutilización de la ip 192.168.0.115 de antigua instalación de OEM Grid Control, así toda la monitorización estaría funcionando hasta el último momento hasta hacer el switch por la nueva.

Se seguirá la guía:


http://www.oracle-base.com/articles/12c/CloudControl12cR1InstallationOnOracleLinux5And6.php

En la creación dbca de una instancia para oem_12c, aparece el error:

ORA-29702: error occurred in Cluster Group Service operation

Se borró el directorio /u01/app/11.2.0/grid después de lanzar el “deinstall -local (que no terminó correctamente)”

Ver enlace:
http://blog.contractoracle.com/2009/08/ora-29702-error-occurred-in-cluster.html



"
cd $ORACLE_HOME/rdbms/lib
make -f ins_rdbms.mk rac_off
make -f ins_rdbms.mk ioracle 

"

Se intenta en nuestro nodo, pero con el siguiente mensaje:

/usr/bin/ld: /u01/app/oracle/product/11.2.0/dbhome_1/lib//libctx11.a(drud.o): don't know how to handle section `' [0x     100]
/u01/app/oracle/product/11.2.0/dbhome_1/lib//libctx11.a: could not read symbols: File format not recognized
collect2: ld devolvió el estado de salida 1
make: *** [/u01/app/oracle/product/11.2.0/dbhome_1/rdbms/lib/oracle] Error 1
 
Se intenta una vez mas el 18/12/2011, con usuario oracle, ubicados en el directorio, y todo correcto, se crean los ejecutables oracle en $ORACLE_HOME/bin y el dbca parece que funciona bien.

Ampliación de espacio para dar cabida a instalación oracle_12c, se añade otro disco de 20GB (dinámico) a la máquina virtualbox y se aplican comandos típicos LVM:

vgdisplay --verbose
fdisk /dev/sdb (n, p, 1, intro, intro, t, 8e (partición Linux VM), w)
partprobe /dev/sdb (para crear el dispositivo /dev/sdb1)
pvcreate /dev/sdb1
vgextend VolGroup00 /dev/sdb1
lvextend -L +10G /dev/VolGroup00/LogVol00
resize2fs /dev/VolGroup00/LogVol00

[root@linux2 software]# df -k
S.ficheros         Bloques de 1K   Usado    Dispon Uso% Montado en
/dev/mapper/VolGroup00-LogVol00
                     27424956  10867584  15141924  42% /
/dev/sda1               101086     19928     75939  21% /boot
tmpfs                   764684         0    764684   0% /dev/shm

Nos quedan 15Gb libres (dejamos 10Gb reservados sin asignar al logVol00 para controlar)

Ahora pasamos los ficheros em12_linux64_disk1of2.zip  y em12_linux64_disk2of2.zip al directorio /u01/software

Seguimos las instrucciones de http://www.oracle-base.com/articles/12c/CloudControl12cR1InstallationOnOracleLinux5And6.php

Nombre de host, cuidado al cambiar valores en /etc/hosts ya que se debe reiniciar el instalador para que coja la nueva configuración de /etc/hosts:

Debe quedar así:
127.0.0.1        linux2.localdomain   linux2

Password administrador: .... (requiere al menos 8 caracteres, siendo 1 numérico)
Datos conexión bbdd: 192.168.0.115 / 1521 / em12c

Revisión instalación, puertos:


Se reciben errores en la parte de configuración que pueden estar relacionados con la lentitud general de la VB, copiamos la VM a otro sistema (Quad-core, 8GB) para ver si conseguimos la instalación completa.
 
Se cambian las rutas del fichero .vbox para que pueda arrancar correctamente desde otra ubicación.

Se le amplia la memoria de la máquina virtual de 1,5Gb a 4Gb y las cpus de 2 a 4, el almacenamiento ahora es local al servidor y no  está montado en un HD USB (antes en el portátil estaba en un HD USB).
 
Se elimina instalación anterior (directorios y entradas en oraInventory/ContentsXML/inventory.xml) incluida las antiguas entradas de RAC:
 
Falla el arranque del listener debido a que quiere notificar a CRS (antigua instalación y no existe), se ve en el listener.log:

“Listener completed notification to CRS on start”

Se arranca correctamente metiendo la siguiente linea en el listener.ora:

SUBSCRIBE_FOR_NODE_DOWN_EVENT_LISTENER=OFF.

Gestión de Xs remotas a través de Xming , para ello, en el servidor:
 
export DISPLAY=192.168.0.197:1.0 (192.168.0.197 es el portátil desde el que lanzaremos la instalación, que se realiza sobre el servidor ubuntu)

Se debe borrar también el esquema SYSMAN_MDS:

<ORACLE_HOME>/sysman/admin/emdrep/bin/RepManager  <HostBasedeDatos> <PuertoListener> <SID/NombreServicio> -action dropall -dbUser <UsuarioBD> -dbPassword <Contraseña BD> -dbRole <Rol Usuario BD> -reposName  <Nombre Repositorio> -reposPassword <ContraseñaRepositorio> -mwHome <MW HOME> -mwOraHome <MW ORA HOME> -oracleHome <ORACLE_HOME>

No conseguimos borrar con este comando, se borran manualmente los usuarios sysman_*, mgmt_view, los sinónimos publicos que apuntaban a ellos.

Una vez se lanza de nuevo el installer:

Fallo en “MDS Schema Configuration”, el los logs aparece:

RCU-6016:Ya existe el prefijo especificado.
RCU-6091:Fallo de validación de prefijo de esquema/nombre de componente.

para corregirlo, se lanza:

“delete from SCHEMA_VERSION_REGISTRY where COMP_NAME='Metadata Services';”,

según post: https://forums.oracle.com/forums/thread.jspa?threadID=1068831

Con ese delete, el “MDS Schema Configuration” se ejecuta correctamente , tras pulsar “reintentar”

Fallo en “Start Oracle Management Service”, se cambian las entradas del host:

127.0.0.1        localhost
192.168.0.115        linux2.localdomain   linux2

Ya que el error aparece en la llamada:

getaddrinfo(localhost , …)  failed (Name or service not known)

Fallo en “Start Oracle Management Service”, ahora parece problema de espacio, se amplia volumen 5GB más:

lvextend -L +5G /dev/VolGroup00/LogVol00
resize2fs /dev/VolGroup00/LogVol00

Y se pulsa sobre reintentar.

Fallo de nuevo en “Start Oracle Management Service”, errores relativos a la lectura del Policy Store, se sigue la nota metalink:

oracle.security.jps.JpsRuntimeException: Cannot Read From Policy Store [ID 1330253.1]

No recuperamos la instalación, se borra todo y se inicia de nuevo, con las siguientes entradas en el /etc/hosts:

192.168.0.115    linux2.localdomain linux2
127.0.0.1       localhost.localdomain localhost

(Borrando los esquemas y tablespaces tb de la bbdd y el registro de la tabla SCHEMA_VERSION_REGISTRY)

Ejecución correcta:

Accesos: https://linux2.localdomain:7802/em
https://linux2.localdomain:7102/console




Usuarios: sysman / ... (clave administrador que introdujimos al principio)
weblogic / welcome (usuario administrador consola weblogic)

Al entrar en "...:7802/em" y logarnos aparece el error:


ADFC-0619: Authorization check failed: 'oracle.jbo.uicli.binding.JUFormDef …

Se observa en el fichero de log: /u01/app/oracle/Middleware/gc_inst/user_projects/domains/GCDomain/servers/EMGC_OMS1/logs/EMGC_OMS1.out

[JpsAuth] Check Permission
         PolicyContext:        [emgc]
         Resource/Target:      [sdk.core.uifwk.template.templateDefPageDef]
         Action:               [view]
         Permission Class:     [oracle.adf.share.security.authorization.RegionPermission]
         Result:               [FAILED]
         For more information on this failure, please set -Djps.auth.debug.verbose=true

Se modifica el fichero jazn-data.xml, situado en : /u01/app/oracle/Middleware/gc_inst/user_projects/domains/GCDomain/servers/EMGC_OMS1/tmp/_WL_user/oracle.security.apm_11.1.1.3.0/p1stos/META-INF
las líneas:

 <class>oracle.security.jps.internal.core.principals.JpsAnonymousRoleImpl</class>
                 <name>anonymous-role</name>

por:

 <class>oracle.security.jps.internal.core.principals.JpsAnonymousRoleImpl</class>
                 <name>Administrators</name>

Ya que el roll anonymous-role no aparece en dicho fichero jazn-data.xml

links:
https://forums.oracle.com/forums/thread.jspa?threadID=2233633
https://forums.oracle.com/forums/thread.jspa?threadID=841951
y bug oracle: Bug 9174870: LOGS INCLUDING A LOT OF ERRORS: [JPSAUTH] CHECK PERMISSION (REGIONPERMISSION)

Se entra en la interfaz de administración (weblogic / welcome)

Se establece la propiedad “-Djps.auth.debug.verbose=true” en los parámetros de arranque de EMGC_OMS1

Se descomenta la parte de jazn-data.xml:

<role-categories>
         <role-category>
           <name>Administrators</name>
           <display-name>Administrators</name>
           <description>Administrators Category</description>
           <members>
             <role-name-ref>APMAdmin</role-name-ref>
           </members>
         </role-category>
       </role-categories>

Sigue apareciendo el mismo error.



Se prueba con https://kr.forums.oracle.com/forums/thread.jspa?threadID=905531

Sin éxito, sigue apareciendo el mismo error.

Se prueba creando usuarios administradores mediante la interfaz de texto emcli, sin éxito.

Pasa otro día, estoy más despejado, ….

Probamos a desabilitar la seguridad ADF según se indica en este post:
https://forums.oracle.com/forums/thread.jspa?threadID=2132032

ENTRAMOS correctamente.

Ficheros modificados:

/u01/app/oracle/Middleware/gc_inst/user_projects/domains/GCDomain/servers/EMGC_OMS1/tmp/_WL_user/oracle.security.apm_11.1.1.3.0/p1stos/adf/META-INF/adf-config.xml
/u01/app/oracle/Middleware/oms/sysman/archives/emgc/deployments/GCDomain/emgc.ear/adf/META-INF/adf-config.xml

Entradas antiguas:
authorizationEnforce="true"
authenticationRequire="true"


Por:

authorizationEnforce="false"
authenticationRequire="false"


En los logs se sigue observanto los errores relativos a los permisos ADF, pero en el frontend ya no se muestra el error ADFC-0619.