[Jc @sdZddkZddkZddkZddklZlZlZlZddk l Z l Z ddk l Z lZddk lZlZddklZdeifd YZd eifd YZd eifd YZdeifdYZdeifdYZdeifdYZ dei!fdYZ"dei#e"fdYZ$dei%fdYZ&dei'fdYZ(dei)fdYZ*de*fdYZ+d ei,fd!YZ-h eei6eei6eei6eei.6eei6eei6e"ei!6e*ei)6e-ei,6e&ei%6e ei6e(ei'6Z/he"d"6e$d#6e"d$6ed%6ed&6ed'6e*d(6e*d)6e&d*6e d+6e+d,6ed-6ed.6e&d/6Z0d0e i1fd1YZ2d2e i3fd3YZ4d4ei5fd5YZ6d6e i7fd7YZ8d8e i9fd9YZ:d:e i;fd;YZ<d<e i=fd=YZ>d>e i?fd?YZ@e4ZAe8eA_Be:eA_Ce<eA_De@eA_Ee>eA_Fe2eA_GdS(@s/Support for the Oracle database. Oracle version 8 through current (11g at the time of this writing) are supported. Driver ------ The Oracle dialect uses the cx_oracle driver, available at http://cx-oracle.sourceforge.net/ . The dialect has several behaviors which are specifically tailored towards compatibility with this module. Connecting ---------- Connecting with create_engine() uses the standard URL approach of ``oracle://user:pass@host:port/dbname[?key=value&key=value...]``. If dbname is present, the host, port, and dbname tokens are converted to a TNS name using the cx_oracle :func:`makedsn()` function. Otherwise, the host token is taken directly as a TNS name. Additional arguments which may be specified either as query string arguments on the URL, or as keyword arguments to :func:`~sqlalchemy.create_engine()` are: * *allow_twophase* - enable two-phase transactions. Defaults to ``True``. * *auto_convert_lobs* - defaults to True, see the section on LOB objects. * *auto_setinputsizes* - the cx_oracle.setinputsizes() call is issued for all bind parameters. This is required for LOB datatypes but can be disabled to reduce overhead. Defaults to ``True``. * *mode* - This is given the string value of SYSDBA or SYSOPER, or alternatively an integer value. This value is only available as a URL query string argument. * *threaded* - enable multithreaded access to cx_oracle connections. Defaults to ``True``. Note that this is the opposite default of cx_oracle itself. * *use_ansi* - Use ANSI JOIN constructs (see the section on Oracle 8). Defaults to ``True``. If ``False``, Oracle-8 compatible constructs are used for joins. * *optimize_limits* - defaults to ``False``. see the section on LIMIT/OFFSET. Auto Increment Behavior ----------------------- SQLAlchemy Table objects which include integer primary keys are usually assumed to have "autoincrementing" behavior, meaning they can generate their own primary key values upon INSERT. Since Oracle has no "autoincrement" feature, SQLAlchemy relies upon sequences to produce these values. With the Oracle dialect, *a sequence must always be explicitly specified to enable autoincrement*. This is divergent with the majority of documentation examples which assume the usage of an autoincrement-capable database. To specify sequences, use the sqlalchemy.schema.Sequence object which is passed to a Column construct:: t = Table('mytable', metadata, Column('id', Integer, Sequence('id_seq'), primary_key=True), Column(...), ... ) This step is also required when using table reflection, i.e. autoload=True:: t = Table('mytable', metadata, Column('id', Integer, Sequence('id_seq'), primary_key=True), autoload=True ) LOB Objects ----------- cx_oracle presents some challenges when fetching LOB objects. A LOB object in a result set is presented by cx_oracle as a cx_oracle.LOB object which has a read() method. By default, SQLAlchemy converts these LOB objects into Python strings. This is for two reasons. First, the LOB object requires an active cursor association, meaning if you were to fetch many rows at once such that cx_oracle had to go back to the database and fetch a new batch of rows, the LOB objects in the already-fetched rows are now unreadable and will raise an error. SQLA "pre-reads" all LOBs so that their data is fetched before further rows are read. The size of a "batch of rows" is controlled by the cursor.arraysize value, which SQLAlchemy defaults to 50 (cx_oracle normally defaults this to one). Secondly, the LOB object is not a standard DBAPI return value so SQLAlchemy seeks to "normalize" the results to look more like other DBAPIs. The conversion of LOB objects by this dialect is unique in SQLAlchemy in that it takes place for all statement executions, even plain string-based statements for which SQLA has no awareness of result typing. This is so that calls like fetchmany() and fetchall() can work in all cases without raising cursor errors. The conversion of LOB in all cases, as well as the "prefetch" of LOB objects, can be disabled using auto_convert_lobs=False. LIMIT/OFFSET Support -------------------- Oracle has no support for the LIMIT or OFFSET keywords. Whereas previous versions of SQLAlchemy used the "ROW NUMBER OVER..." construct to simulate LIMIT/OFFSET, SQLAlchemy 0.5 now uses a wrapped subquery approach in conjunction with ROWNUM. The exact methodology is taken from http://www.oracle.com/technology/oramag/oracle/06-sep/o56asktom.html . Note that the "FIRST ROWS()" optimization keyword mentioned is not used by default, as the user community felt this was stepping into the bounds of optimization that is better left on the DBA side, but this prefix can be added by enabling the optimize_limits=True flag on create_engine(). Two Phase Transaction Support ----------------------------- Two Phase transactions are implemented using XA transactions. Success has been reported of them working successfully but this should be regarded as an experimental feature. Oracle 8 Compatibility ---------------------- When using Oracle 8, a "use_ansi=False" flag is available which converts all JOIN phrases into the WHERE clause, and in the case of LEFT OUTER JOIN makes use of Oracle's (+) operator. Synonym/DBLINK Reflection ------------------------- When using reflection with Table objects, the dialect can optionally search for tables indicated by synonyms that reference DBLINK-ed tables by passing the flag oracle_resolve_synonyms=True as a keyword argument to the Table construct. If DBLINK is not in use this flag should be left off. iN(tutiltsqltschematlog(tdefaulttbase(tcompilertvisitors(t operatorst functions(ttypest OracleNumericcBseZdZRS(cCs5|idjodSdh|id6|id6SdS(NtNUMERICs!NUMERIC(%(precision)s, %(scale)s)t precisiontscale(R tNoneR(tself((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyt get_col_specs(t__name__t __module__R(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR st OracleIntegercBseZdZRS(cCsdS(NtINTEGER((R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRs(RRR(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRstOracleSmallIntegercBseZdZRS(cCsdS(NtSMALLINT((R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRs(RRR(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRst OracleDatecBs#eZdZdZdZRS(cCsdS(NtDATE((R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRscCsdS(N(R(Rtdialect((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pytbind_processorscCs d}|S(NcSs&t|tip|S|iSdS(N(t isinstancetdatetimetdate(tvalue((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pytprocesss((RRR ((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pytresult_processors (RRRRR!(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRs  tOracleDateTimecBseZdZdZRS(cCsdS(NR((R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRscCs d}|S(NcSsW|djpt|tio|Sti|i|i|i|i|i|iSdS(N( RRRtyeartmonthtdaythourtminutetsecond(R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR s ((RRR ((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR!s (RRRR!(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR"s tOracleTimestampcBs#eZdZdZdZRS(cCsdS(Nt TIMESTAMP((R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRscCs|iS(N(R*(RR((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pytget_dbapi_typescCs d}|S(NcSsW|djpt|tio|Sti|i|i|i|i|i|iSdS(N( RRRR#R$R%R&R'R((R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR s ((RRR ((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR!s (RRRR+R!(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR)s  t OracleStringcBseZdZRS(cCsdh|id6S(NsVARCHAR(%(length)s)tlength(R-(R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRs(RRR(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR,stOracleNVarcharcBseZdZRS(cCsdh|id6S(NsNVARCHAR2(%(length)s)R-(R-(R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRs(RRR(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR.st OracleTextcBs#eZdZdZdZRS(cCs|iS(N(tCLOB(Rtdbapi((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR+scCsdS(NR0((R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRscsItt|i||ipS|iifd}|S(NcsPt|o&o|iS|iSno |S|SdS(N(Rtread(R(tlobt super_process(s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR s (tsuperR/R!tauto_convert_lobsR1tLOB(RRR ((R3R4s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR!s    (RRR+RR!(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR/s  t OracleCharcBseZdZRS(cCsdh|id6S(NsCHAR(%(length)s)R-(R-(R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRs(RRR(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR8st OracleBinarycBs,eZdZdZdZdZRS(cCs|iS(N(tBLOB(RR1((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR+scCsdS(NR:((R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRscCsdS(N(R(RR((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRscs.|ipdS|iifd}|S(Ncs#t|o |iS|SdS(N(RR2(R(R3(s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR s (R6RR1R7(RRR ((R3s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR!s   (RRR+RRR!(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR9s   t OracleRawcBseZdZRS(cCsdh|id6S(NsRAW(%(length)s)R-(R-(R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRs(RRR(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR;st OracleBooleancBs#eZdZdZdZRS(cCsdS(NR((R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR scCs d}|S(NcSs$|djodS|otptS(N(RtTruetFalse(R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR s ((RRR ((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR! s cCs d}|S(NcSsL|tjodS|tjodS|djodS|otptSdS(Nii(R=R>R(R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR s   ((RRR ((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRs (RRRR!R(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR<s  tVARCHAR2t NVARCHAR2tCHARRtDATETIMEtNUMBERR:tBFILER0R*tRAWtFLOATsDOUBLE PRECISIONtLONGtOracleExecutionContextcBs#eZdZdZdZRS(cCs$tt|i|iio|in|idj ot|idjox|i i D]}|i i |}|i i |}|id|}|i ov|i i|ii|ii}t|dp h|_n|ii||i|<|i||id|}||jo+ti||tt||||qqW|ioH|i}|ot|}nd}|i i |i ||i}n |i }td|i d|i d |d|id |i}d |ijo|id |d RRRR[RRRR=RRR Rt ischema_namesR,tretsubtKeyErrorRtwarnRtNULLTYPERt DefaultClauseRttextt append_columntColumntcolumnsRttuplet primary_keytaddR`RtTabletmetadataRbtappend_constrainttForeignKeyConstraint('RR^ttabletinclude_columnstpreparertresolve_synonymst actual_nameRRtsynonymR`RtcolnametcoltypeR-R RRRtcolargstfksR}txt cons_namet cons_typet local_columnt remote_tablet remote_columnt remote_ownertfktref_remote_nametref_remote_ownert ref_dblinkt ref_synonymtrefspecttt_[2]R[R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyt reflecttable8s C &+  [     #' "  Q    9!1>3 N(sdialectR(%RRR[R=tsupports_alterR>tsupports_unicode_statementstmax_identifier_lengthtsupports_sane_rowcounttsupports_sane_multi_rowcounttpreexecute_pk_sequencestsupports_pk_autoincrementtdefault_paramstyleRrR1t classmethodRRRRRRRRRRRRRRRRRtconnection_memoizeRRR(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRmms@   )            *t_OuterJoinColumncBseZdZdZRS(touter_join_columncCs ||_dS(N(Rj(RRj((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRrs(RRt__visit_name__Rr(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRstOracleCompilercBseZdZeiiiZeihdei 6dei 6eii iZ e ihde i 6dZdZdZdZdZd Zd Zed Zd Zd ZdZdZRS(sOracle compiler modifies the lexical structure of Select statements to work under non-ANSI configured Oracle databases, if the use_ansi flag is False. cCsd||fS(s mod(%s, %s)((Rty((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pytscCsd||fS(sCONTAINS (%s, %s)((RR"((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR#stCURRENT_TIMESTAMPcOs&tt|i||h|_dS(N(R5R!Rrt_OracleCompiler__wheres(RtargsR|((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRrscCsdS(sCalled when a ``SELECT`` statement has no froms, and no ``FROM`` clause is to be appended. The Oracle compiler tacks a "FROM DUAL" to the statement. s FROM DUAL((R((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyt default_fromscCst|idjS(Ni(RNR(Rtfunc((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pytapply_function_parensscKsX|iiotii|||S|i|idtd|i|idtSdS(Ntasfroms, ( RRsRtDefaultCompilert visit_joinR tleftR=tright(RRR|((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR,s csMgfd}x(|D] }ti|hh|d6qWtiS(NcsWio9fd}itiihh|d6niidS(Ncss|itijo\|iiijot|i|_qo|iiijot|i|_qondS(N(toperatort sql_operatorsteqR-RR.R(tbinary(R(s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyt visit_binarys R2(tisouterRRtcloned_traversetonclause(RR3(R(Rs?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR,s *R(RttraverseRtand_(RtfromsR,tf((Rs?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyt_get_nonansi_join_whereclauses  cCs|i|idS(Ns(+)(R Rj(Rtvc((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pytvisit_outer_join_columnscCs|iii|dS(Ns.nextval(RRtformat_sequence(Rtseq((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pytvisit_sequencescKs[|o=|i|id||d|ii||i|iS|i|i|SdS(sFOracle doesn't like ``FROM table AS alias``. Is the AS standard SQL??R*t N(R toriginalRt format_aliast _anonymizeR[(RtaliasR*R|((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyt visit_aliass=cCsdS(sNNeed to determine how to get ``LIMIT``/``OFFSET`` into a ``UNION`` for Oracle.N((Rtselect((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyt_TODO_visit_compound_selectsc Kst|dd pW|iip~|io)d|idjo|idd}nd }|i|}|i|}|o|i|}t|_ qn|i d j p|i d j o|i }t|_ t ig}|iD] }||q~}|i o'|iio|id|i }nt|_ t|_|i d j oJ|i } |i d j o| |i 7} n|it id| jn|i d jo |}qf|it idid}t|_ t|_t ig} |iD]!}|idjo | |qq~ } t| _ t| _| it id|i j| }qjnt|dt|dRR+t visit_select( RRGR|t existingfromsR9t whereclauseR}R`t limitselecttmax_rowRt offsetselect((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRY sN   -      ;  cCsdS(NR((RRG((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyt limit_clauseSscCs/|idjodStt|i|SdS(Ntnowaits FOR UPDATE NOWAIT(t for_updateR5R!tfor_update_clause(RRG((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRbVs(RRt__doc__RR+RtcopytupdateR0tmodtmatch_opR t sql_functionstnowRrR'R)R,R;R=R@R>RFRHRYR_Rb(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR!s,           G tOracleSchemaGeneratorcBseZdZdZRS(cKs|ii|}|d|ii|ii7}|i|}|dj o|d|7}n|ip|d7}n|S(NRAs DEFAULT s NOT NULL( Rt format_columnRSRTRRtget_column_default_stringRR(RRjR|tcolspecR((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pytget_column_specification^s#  cCs]|i p#|ii|i|i|i o+|id|ii||i ndS(NsCREATE SEQUENCE %s( t checkfirstRRR^R[RRRR>R(Rtsequence((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR@is.(RRRnR@(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRj]s tOracleSchemaDroppercBseZdZRS(cCs\|i p"|ii|i|i|io+|id|ii||i ndS(NsDROP SEQUENCE %s( RoRRR^R[RRRR>R(RRp((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR@os-(RRR@(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRqnstOracleDefaultRunnercBseZdZRS(cCs'|id|iii|dhS(NsSELECT s.nextval FROM DUAL(texecute_stringRRR>(RR?((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyR@us(RRR@(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRrtstOracleIdentifierPreparercBseZdZRS(cCs1tidd|i}tt|i||S(Ns^_+R(RRtidentR5Rttformat_savepoint(Rt savepointR[((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRvys(RRRv(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyRtxs(HRcRRRt sqlalchemyRRRRtsqlalchemy.engineRRtsqlalchemy.sqlRRRR0R RhR RtNumericR tIntegerRt SmallintegerRtDateRtDateTimeR"R*R)tStringR,tUnicodeR.tTextR/RAR8tBinaryR9R;tBooleanR<tFloatRRtDefaultExecutionContextRHRqRmt ClauseElementRR+R!tSchemaGeneratorRjt SchemaDropperRqt DefaultRunnerRrtIdentifierPreparerRtRtstatement_compilertschemageneratort schemadropperRt defaultrunnertexecution_ctx_cls(((s?/usr/lib/python2.6/site-packages/sqlalchemy/databases/oracle.pyt|s~$"             .K