четверг, 13 ноября 2014 г.

Stream таблиц

создать пользователей STREAM_ADMIN на обоих базах

DECLARE
tables varchar2(4000);
BEGIN
tables := '"SHM1"."TABLE1",
"SHM1"."TABLE2"';
DBMS_STREAMS_ADM.MAINTAIN_TABLES(
table_names   => tables,
source_directory_object      =>      'DTPUMP',
destination_directory_object     =>      'DTPUMP',
source_database          =>      'src.domain',
destination_database         =>      target.domain',
perform_actions              =>      TRUE,
script_name                      =>      'streams_setup.sql',
script_directory_object          =>      'DTPUMP',
dump_file_name               =>      'streams_setup.dmp',
capture_queue_user           =>       '"STREAM_ADMIN"',
apply_queue_user             =>      '"STREAM_ADMIN"',
log_file                 =>      'streams_setup.log',
bi_directional               =>      false,
include_ddl                  =>      true);
END;
/


на исходной базе пронать инициализацию
BEGIN
for rec in (select distinct schema_name||'.'||object_name  as tbl from ALL_STREAMS_RULES)
loop
DBMS_CAPTURE_ADM.PREPARE_TABLE_INSTANTIATION (
  table_name   => rec.tbl
, supplemental_logging => 'all'
);
end loop;
END;
/

<scn>=select DBMS_FLASHBACK.GET_SYSTEM_CHANGE_NUMBER from dual


на таргетной тоже, scn взять с исходной

BEGIN
for rec in (select distinct schema_name||'.'||object_name  as tbl from ALL_STREAMS_RULES)
loop
DBMS_APPLY_ADM.SET_TABLE_INSTANTIATION_SCN (
  source_object_name   => rec.tbl
, source_database_name => 'target.domain'
, instantiation_scn    => <scn>
);
end loop;
END;
/



создание STREAM ADMIN

CREATE TABLESPACE STREAM_TBS DATAFILE '+ASMDATA' 
  SIZE 1G REUSE AUTOEXTEND ON MAXSIZE UNLIMITED;

CREATE USER stream_admin IDENTIFIED BY <password>
   DEFAULT TABLESPACE stream_tbs
   QUOTA UNLIMITED ON stream_tbs;


GRANT CONNECT, RESOURCE, DBA TO stream_admin;

BEGIN
  DBMS_STREAMS_AUTH.GRANT_ADMIN_PRIVILEGE(
    grantee          => 'stream_admin',    
    grant_privileges => true);
END;
/


COMMIT
/

четверг, 9 октября 2014 г.

SMTP, отсылка писем

С учетом того, что smtp системы перешли на SSL/TLS
оракл необходимо поднастроить с учетом этого

1. создать wallet
  mkdir /opt/oracle/wallet
  mkstore -wrl "/opt/oracle/wallet" -create

2) посмотреть корневой сертификат вашего смтп сервера
openssl s_client -connect smtp.yandex.ru:587 -starttls smtp

CONNECTED(00000003)
depth=2 C = PL, O = Unizeto Sp. z o.o., CN = Certum CA
verify return:1
depth=1 C = PL, O = Unizeto Technologies S.A., OU = Certum Certification Authority, CN = Certum Level IV CA
verify return:1
depth=0 C = RU, O = Yandex LLC, OU = ITO, L = Moscow, ST = Russian Federation, CN = smtp.yandex.ru, emailAddress = pki@yandex-team.ru
verify return:1

закачать этот сертификат себе
и записать в wallet

orapki wallet add -wallet /opt/oracle/wallet/ewallet.p12 -trusted_cert -cert unizeto.crt


3) Не забыть настроить ACL для соотствующих портов

BEGIN
  DBMS_NETWORK_ACL_ADMIN.assign_acl (
    acl         => 'acl_smtp.xml',
    host        => 'smtp.yandex.ru', 
    lower_port  => 587,
    upper_port  => 587);
    COMMIT;
END;

просмотреть чего там в ACL настроено можно так
SELECT host, lower_port, upper_port, acl
FROM   dba_network_acls;

SELECT * FROM dba_network_acl_privileges;


4) ну и цепляемся к серверу

       l_reply := UTL_SMTP.OPEN_CONNECTION(
         host => common.GET_SETTING('MAILSERVER',MAILSERVER),
         port => common.GET_SETTING('MAILPORT',MAILPORT),
         c => l_mail_conn,
         wallet_path => 'file:/opt/oracle/wallet',
         wallet_password => 'wallet_pas',
         secure_connection_before_smtp => FALSE
         );
    IF l_reply.code != 220
      THEN
        raise_application_error(-20000, 'utl_smtp.open_connection: '||l_reply.code||' - '||l_reply.text);
      END IF;
  dbms_output.put_line('utl_smtp.ehlo');

  l_replies := utl_smtp.ehlo(l_mail_conn, 'pmga.ru');

  FOR ri IN 1..l_replies.COUNT
  LOOP
    dbms_output.put_line(l_replies(ri).code||' - '||l_replies(ri).text);
  END LOOP;
        
      l_reply := UTL_SMTP.STARTTLS(l_mail_conn);
  IF l_reply.code != 220
  THEN
    raise_application_error(-20000, 'utl_smtp.starttls: '||l_reply.code||' - '||l_reply.text);
  END IF;      


      l_reply := UTL_SMTP.AUTH(
      c => l_mail_conn,
      username => common.GET_SETTING('MAILLOGIN','pmgadmin'),
      password => common.GET_SETTING('MAILPASS','pwd13pmg'),
      schemes  => utl_smtp.all_schemes);

 IF l_reply.code != 235
  THEN
    raise_application_error(-20000, 'utl_smtp.auth: '||l_reply.code||' - '||l_reply.text);
  END IF;

А дальше стандартное общение

      UTL_SMTP.MAIL(l_mail_conn,  common.GET_SETTING('MAILLOGIN',SENDER_EMAIL));
      UTL_SMTP.RCPT(l_mail_conn, in_mailto);

      UTL_SMTP.OPEN_DATA(l_mail_conn); -- open data sheet
      UTL_SMTP.WRITE_DATA(l_mail_conn, l_data);

      UTL_SMTP.CLOSE_DATA(l_mail_conn);
      UTL_SMTP.QUIT(l_mail_conn);


вторник, 30 сентября 2014 г.

RAC , RMAN и netbackup

Исходные данные: 2 ноды RAC (ora-big1, ora-big2), сервер netbackup (vm-netbackup)

Задача: настроить корректную схему бэкапа - incr 0 (1/month), cumm (1/week), incr 1 (1/day)
Скрипты бэкапов:
::::::::::::::
/opt/oracle-shared/backup/arch_backup.rman
::::::::::::::
backup device type sbt archivelog all delete all input;

::::::::::::::
/opt/oracle-shared/backup/cumm_backup.rman
::::::::::::::
backup incremental level 1 CUMULATIVE database plus archivelog delete input;
delete noprompt backup of archivelog until time 'sysdate-1';
crosscheck backup;
#delete noprompt obsolete;

::::::::::::::
/opt/oracle-shared/backup/full_backup.rman
::::::::::::::
crosscheck archivelog all;
crosscheck backup;
backup incremental level 0 database plus archivelog delete input;
#delete noprompt obsolete ;

::::::::::::::
/opt/oracle-shared/backup/incr_backup.rman
::::::::::::::
backup incremental level 1 database plus archivelog delete input;
delete noprompt backup of archivelog until time 'sysdate-1';
crosscheck backup;
#delete noprompt obsolete ;

Проблема: по умолчанию ноды не видят бэкапы сделанные другой нодой. соответственно crosscheck уводит в expired бэкапа сделанные другой нодой, а delete nopromp obsolete затирает их


Решение:

в rman:
CONFIGURE DEVICE TYPE 'SBT_TAPE' PARALLELISM 2 BACKUP TYPE TO BACKUPSET;
CONFIGURE CHANNEL 1 DEVICE TYPE 'SBT_TAPE' PARMS  'ENV=(NB_ORA_POLICY=oraPOLICY,NB_ORA_SERV=vm-netbackup,NB_ORA_CLIENT=ora-big1)';
CONFIGURE CHANNEL 2 DEVICE TYPE 'SBT_TAPE' PARMS  'ENV=(NB_ORA_POLICY=oraPOLICY,NB_ORA_SERV=vm-netbackup,NB_ORA_CLIENT=ora-big2)';

на vm-netbackup
cd /usr/opnv/netbackup/db/altnames
echo "ora-big2" >> ora-big1
echo "ora-big1" >> ora-big2

пятница, 19 сентября 2014 г.

смена пароля у ASMSNMP

$ asmcmd
ASMCMD> lspwusr
Username sysdba sysoper sysasm
     SYS   TRUE    TRUE   TRUE
 ASMSNMP   TRUE   FALSE  FALSE
ASMCMD>  orapwusr --modify --password ASMSNMP
Enter password: **********
ASMCMD>

четверг, 18 сентября 2014 г.

enq: TM - contention

Ожидание enq: TM - contention проявляется при наличии неиндексированных внешних ключей


Поиск всех нединдексированных полей в схеме

SELECT * FROM (
SELECT c.table_name, cc.column_name, cc.position column_position
FROM   user_constraints c, user_cons_columns cc
WHERE  c.constraint_name = cc.constraint_name
AND    c.constraint_type = 'R'
MINUS
SELECT i.table_name, ic.column_name, ic.column_position
FROM   user_indexes i, user_ind_columns ic
WHERE  i.index_name = ic.index_name
)
ORDER BY table_name, column_position;

вторник, 9 сентября 2014 г.

Corrupted blocks


подробнее про всякие методики решения проблемы
http://www.oracle-base.com/articles/misc/detect-and-correct-corruption.php


если база не в архивлог режиме, и если не ентрепрайз - то подручными методами

Поиск объекта с битым блоком

SELECT SEGMENT_TYPE,OWNER||'.'||SEGMENT_NAME
   FROM DBA_EXTENTS
   WHERE FILE_ID = <file_id> AND <corrupted_block> BETWEEN BLOCK_ID
   AND BLOCK_ID+BLOCKS -1

далее решаем

если индекс - пересоздаем - все просто
если таблица - пробуем либо оттранкетить либо пересоздать



среда, 16 июля 2014 г.

Копирование архивлогов на стендбай

вытаскиваем  архивлоги из асм в файловую систему

for i in $(asmcmd ls +ASMREDO/NAUCRM/ARCHIVELOG/); do
mkdir /opt/oracle/arc/$i
 for j in $(asmcmd ls +ASMREDO/NAUCRM/ARCHIVELOG/$i); do

  asmcmd cp +ASMREDO/NAUCRM/ARCHIVELOG/$i/$j /opt/oracle/arc/$i
done
done


перекидываем по scp на стендбай
регистрируем оптом
rman target /
catalog start with '/opt/oracle/arc'; 
recover standby database until cancel;

понедельник, 9 июня 2014 г.

четверг, 29 мая 2014 г.

Использование типа Object

Пример организации циклического буфера FIFO

Создаем глобальный тип массива на 100 элементов

CREATE type t_last_sess_arr is varray(100) of number(32,0);

Создаем объектный тип с полями текущей позиции массива и самим массивом.
В объекте 3 метода

  1. конструктор, инициализирующий массив пустыми значениями 
  2. процедура которая сдвигает позицию и заменяет элемент массива в этой позиции
  3. функция, которая возвращает объект с обновленным элементом и сдвинутой позицией
CREATE OR REPLACE
TYPE t_last_sess   AS OBJECT (
         position       INT
        , last_sess    t_last_sess_arr
        , CONSTRUCTOR FUNCTION t_last_sess
                RETURN SELF AS RESULT
        , MEMBER PROCEDURE addBS (b_session_id NUMBER)
        , MEMBER FUNCTION updArr (b_session_id NUMBER) RETURN t_last_sess
   );
/

CREATE OR REPLACE
TYPE BODY t_last_sess AS
  CONSTRUCTOR FUNCTION t_last_sess
    RETURN SELF AS RESULT
  AS
  BEGIN
    SELF.position := 1;
    SELF.last_sess := t_last_sess_arr();
    SELF.last_sess.EXTEND(100);
    RETURN;
  END t_last_sess;
  MEMBER PROCEDURE addBS (b_session_id NUMBER)
  IS
  BEGIN
    SELF.last_sess(SELF.position) := b_session_id;
    IF SELF.position = 100 THEN
        SELF.position := 1;
    ELSE
        SELF.position := SELF.position +1;
    END IF;
    RETURN;
  END addBS;
  MEMBER FUNCTION updArr (b_session_id NUMBER) RETURN t_last_sess
  IS
    l_arr t_last_sess;
  BEGIN
    l_arr:= SELF;
    l_arr.addBS(b_session_id);
    RETURN l_arr;
  END updArr;
END;
/



Пример использования данного объектного типа

Буфер для сохранения последних 100 сессий по каждому отдельному типу
create table last_sessions (
type_id int primary key,
sess_arr t_last_sess)
organization index;

Инициализация буфера для типа сессий=1
INSERT INTO last_sessions VALUES(1, t_last_sess());

Обновление буфера
update last_sessions t set t.sess_arr=t.sess_arr.updArr(<ИД сессии>) where bill_type_id=1;

среда, 14 мая 2014 г.

Трассировка shared сессий

Определяем сессию для трассировки - SID, SERIAL# и инстанс в случае RAC

Трассировка по шаредам складывается в трейс файлы диспетчеров  и шаред серверов
Чтобы не собирать старую историю можно удалить(перенести) все трейс файлы из USER_DUMP_DEST

Включаем трассировку
exec dbms_monitor.SESSION_TRACE_ENABLE(SID, SERIAL#, TRUE, TRUE);

Выключаем трассировку
exec dbms_monitor.SESSION_TRACE_DISABLE(SID);

собираем данные из кучи файлов
trcsess output=trace.log session=SID.SERIAL# *

прогоняем tkprof
tkprof trace.log

Изучаем и анализируем результат

пятница, 11 апреля 2014 г.

добавления файла в стэндбай

В случае когда добавляются файлы в основную базу. На стэндбае они не появляются и автоматическая накатка архивлогов останавливается. Если руками попробовать восстановить мы получим следующую ошибку:
ORA-00283: recovery session canceled due to errors
ORA-01111: name for data file 6 is unknown - rename to correct file
ORA-01110: data file 6: '/opt/oracle/product/11gR2/db/dbs/UNNAMED00006'
ORA-01157: cannot identify/lock data file 6 - see DBWR trace file
ORA-01111: name for data file 6 is unknown - rename to correct file
ORA-01110: data file 6: '/opt/oracle/product/11gR2/db/dbs/UNNAMED00006'


На standby базе данных

1.select name from v$datafile;
2. видим что то типа /opt/oracle/product/11gR2/db/dbs/UNNAMED00006
3. выполняем команду

alter database create datafile '/opt/oracle/product/11gR2/db/dbs/UNNAMED00006' as '+DATA' (или ваша нужная локация)

4.проверяем select name from v$datafile;
5. запускаем ручной рековер recover standby database until cancel;
6. переводим в авторежим alter database recover managed standby database ;

понедельник, 31 марта 2014 г.

ora-00845 memory_target not supported on this system ubuntu 13.10

При установке oracle-xe  на убунту может возникнуть такая замечательная ошибка.
Связана она с тем, что при установленном параметре memory_target, а по умолчанию он установлен, оракл хочет tmpfs замонтированный в /dev/shm, а убунта монтирует его в /run/shm, и делает симлинк /dev/shm на /run/shm.
Оракл так работать не хочет.
Монтирование tmpfs у убунты  прописано в хитром месте /lib/init/fstab
а линк в /etc/init/mounted-dev.conf

в итоге после редактирования файлы стали выглядеть следующим образом
/lib/init/fstab
# /lib/init/fstab: static file system information.
#
# These are the filesystems that are always mounted on boot, you can
# override any of these by copying the appropriate line from this file into
# /etc/fstab and tweaking it as you see fit.  See fstab(5).
#
# <file system> <mount point>             <type>          <options>                               <dump> <pass>
/dev/root       /                         rootfs          defaults                                     0 1
none            /proc                     proc            nodev,noexec,nosuid                          0 0
none            /proc/sys/fs/binfmt_misc  binfmt_misc     nodev,noexec,nosuid,optional                 0 0
none            /sys                      sysfs           nodev,noexec,nosuid                          0 0
none            /sys/fs/cgroup            tmpfs           optional,uid=0,gid=0,mode=0755,size=1024     0 0
none            /sys/fs/fuse/connections  fusectl         optional                                     0 0
none            /sys/kernel/debug         debugfs         optional                                     0 0
none            /sys/kernel/security      securityfs      optional                                     0 0
none            /sys/firmware/efi/efivars efivarfs        optional                                     0 0
none            /spu                      spufs           gid=spu,optional                             0 0
none            /dev                      devtmpfs,tmpfs  mode=0755                                    0 0
none            /dev/pts                  devpts          noexec,nosuid,gid=tty,mode=0620              0 0
none            /tmp                      none            defaults                                     0 0
none            /run                      tmpfs           noexec,nosuid,size=10%,mode=0755             0 0
none            /run/lock                 tmpfs           nodev,noexec,nosuid,size=5242880             0 0
none            /dev/shm                  tmpfs           nosuid,nodev                                 0 0
none            /run/user                 tmpfs           nodev,noexec,nosuid,size=104857600,mode=0755 0 0
none            /sys/fs/pstore            pstore          optional                                     0 0


/etc/init/mounted-dev.conf
# mounted-dev - Populate /dev filesystem
#
# Populates the /dev filesystem from /lib/udev/devices once the temporary
# filesystem mount is in place.

description     "Populate /dev filesystem"

start on mounted MOUNTPOINT=/dev
env MOUNTPOINT=/dev
env container
env LIBVIRT_LXC_UUID

task

script
    # With the /run transition, shm lives there now, but eglibc still looks in
    # /dev/shm. So create a symlink there
    [ -e /run/shm ] || ln -s /dev/shm /run/shm
   
    if [ "${TYPE}" != "devtmpfs" ]
    then
        cd "${MOUNTPOINT}"
        /sbin/MAKEDEV std fd ppp tun
        # lxc and libvirt containers have set up their own console and ttys
        if [ -z "$container" -a -z "$LIBVIRT_LXC_UUID" ]; then
                /sbin/MAKEDEV console
        fi
    fi

end script

Настройка диагностических сообщение в oracle 11g

Если обнаруживаете вдруг, что утекает место - проверяйте диагностические файлы


По умолчанию лежит в $ORACLE_BASE/diag

управляется утилитой adrci
по умолчанию автоудаление выставлено на 30 дней и год

adrci> show homes
ADR Homes:
diag/rdbms/lotosgm/lotosgm
diag/rdbms/streetco/streetco
diag/tnslsnr/oracle11/listener
diag/clients/user_oracle/host_1956530261_11
diag/clients/user_oracle/host_1956530261_80


adrci> set home diag/rdbms/streetco/streetco  -указываем хоме, с которым хотим работать


adrci> show control

ADR Home = /opt/oracle/database/diag/rdbms/streetco/streetco:
*************************************************************************
ADRID                SHORTP_POLICY        LONGP_POLICY         LAST_MOD_TIME                            LAST_AUTOPRG_TIME                        LAST_MANUPRG_TIME                        ADRDIR_VERSION       ADRSCHM_VERSION      ADRSCHMV_SUMMARY     ADRALERT_VERSION     CREATE_TIME                            
-------------------- -------------------- -------------------- ---------------------------------------- ---------------------------------------- ---------------------------------------- -------------------- -------------------- -------------------- -------------------- ----------------------------------------
1800226057           168                  168                  2014-03-31 13:38:09.541128 +04:00        2014-03-27 23:05:35.270114 +04:00                                                 1                    2                    80                   1                    2014-03-25 21:51:13.935660 +04:00    
1 rows fetched


adrci>  set control (SHORTP_POLICY=168)
adrci> set control (LONGP_POLICY=168)

выставляем политики

adrci> show alert - смотрим алерты

пятница, 14 марта 2014 г.

Multipath для ASM - нюансы

Нижеописанная настройка позволяет легко оперировать дисковыми устройствами выделенными для ASM. 
Привязываем WWID диска к имени
Через udev указываем права на устройство с данным именем

/etc/multipath.conf


defaults {
        find_multipaths yes
        user_friendly_names yes
}
multipaths {
    multipath {
        wwid                    36782bcb0005b6a48000012e9531821da
        alias                   oraVD
    }
}

/etc/udev/rules.d/50-dm.rules 

ENV{DM_NAME}=="oraVD*", OWNER:="oracle", GROUP:="dba", MODE:="660", SYMLINK+="mapper/$env{DM_NAME}"


multipath -r - сканирование устройств