Quantcast
Channel: Teradata Downloads - Connectivity
Viewing all articles
Browse latest Browse all 445

Getting both Teradata ODBC and Aster ODBC working together with the pyodbc Python adapter

$
0
0

Greetings,
I'm generally more of a forum lurker, but every once in a while I get something working that doesn't readily work out of the box the way you'd expect and feel that its worth posting about to ease the path of others (and maybe leave a trail for Google to scan so I don't have to remember where I left my notes!)
My little victory dance today was getting the ODBC adapters for Aster and Teradata to both co-exist in the same process with pyodbc, the Python ODBC adapter.  It took more doing than it should have, so thats why I'm writing it up.
My test system was Ubuntu Linux 13.04 -- a very recent Ubuntu build -- running a 64 bit Linux.
I had to install the "rpm" module for Ubuntu to cope with the tdodbc RPMs, but that's not a big deal.  The teradata ODBC adapters are in their typical home of

/opt/teradata/client/14.10/odbc_64/lib

and

/opt/teradata/client/14.10/tdicu/lib64

and to make things similar, I dropped the Aster ODBC drivers into /opt/aster, i.e. the two drivers are

/opt/aster/stage/clients-odbc-linux64/Libs

and

/opt/aster/stage/clients-odbc-linux64/DataDirect/lib

Python is the stock python and I used the Ubuntu package manager to install pyodbc.
Now -- heres the interesting stuff.  The Teradata installer will make symlinks to libodbc.so and libodbcinst.so in /usr/lib64.  Fortunately for me, /usr/lib64 is not a standard library path on Ubuntu so I could easily see what was going on.  The Teradata version of the DataDirect odbc driver manager will not properly load the Aster version of the adapter, so you have to switch out the symlinks in /usr/lib64.  First, you have to make libodbc.so and libodbcinst.so use the Aster versions, and you also need to make a convenience link of libodbc.so.1 for pydobc itself.   If there are existing libodbc* in /usr/lib64 (or, for me, /lib/x86_64-linux-gnu) you need to get those out of the way -- I suggest renaming them with a capital X in front so they aren't found but you can easily put them back in case you oops something.
You also have to symlink the lidddicu27.so from Aster into your lib directory.  Here's the symlinks I ended up with when I was done (note that the teradata odbc installer did most of these links).

lrwxrwxrwx root/root         0 2013-11-06 14:17 usr/lib64/libodbc.so.1 -> libodbc.so
lrwxrwxrwx root/root         0 2013-11-06 14:02 usr/lib64/libodbc.so -> /opt/aster/stage/clients-odbc-linux64/DataDirect/lib/libodbc.so
lrwxrwxrwx root/root         0 2013-11-06 14:02 usr/lib64/libodbcinst.so -> /opt/aster/stage/clients-odbc-linux64/DataDirect/lib/libodbcinst.so
lrwxrwxrwx root/root         0 2013-11-06 10:29 usr/lib64/libiculxtd.so -> /opt/teradata/client/14.10/tdicu/lib64/libiculxtd.so.46.0
lrwxrwxrwx root/root         0 2013-11-06 10:29 usr/lib64/libicui18ntd.so -> /opt/teradata/client/14.10/tdicu/lib64/libicui18ntd.so.46.0
lrwxrwxrwx root/root         0 2013-11-06 10:29 usr/lib64/libicudatatd.so.46 -> /opt/teradata/client/14.10/tdicu/lib64/libicudatatd.so.46.0
lrwxrwxrwx root/root         0 2013-11-06 10:29 usr/lib64/libiculetd.so -> /opt/teradata/client/14.10/tdicu/lib64/libiculetd.so.46.0
lrwxrwxrwx root/root         0 2013-11-06 10:29 usr/lib64/libtdsso.so -> /opt/teradata/client/ODBC_64/lib/libtdsso.so
lrwxrwxrwx root/root         0 2013-11-06 10:29 usr/lib64/libicudatatd.so -> /opt/teradata/client/14.10/tdicu/lib64/libicudatatd.so.46.0
lrwxrwxrwx root/root         0 2013-11-06 10:29 usr/lib64/libtdparse.so -> /opt/teradata/client/ODBC_64/lib/libtdparse.so
lrwxrwxrwx root/root         0 2013-11-06 10:29 usr/lib64/libicuuctd.so -> /opt/teradata/client/14.10/tdicu/lib64/libicuuctd.so.46.0
lrwxrwxrwx root/root         0 2013-11-06 10:29 usr/lib64/libicuuctd.so.46 -> /opt/teradata/client/14.10/tdicu/lib64/libicuuctd.so.46.0
lrwxrwxrwx root/root         0 2013-11-06 10:29 usr/lib64/libicui18ntd.so.46 -> /opt/teradata/client/14.10/tdicu/lib64/libicui18ntd.so.46.0
lrwxrwxrwx root/root         0 2013-11-06 10:29 usr/lib64/libicuiotd.so -> /opt/teradata/client/14.10/tdicu/lib64/libicuiotd.so.46.0
lrwxrwxrwx root/root         0 2013-11-06 10:29 usr/lib64/libicuiotd.so.46 -> /opt/teradata/client/14.10/tdicu/lib64/libicuiotd.so.46.0
lrwxrwxrwx root/root         0 2013-11-06 10:29 usr/lib64/libddicu26.so -> /opt/teradata/client/ODBC_64/lib/libddicu26.so
lrwxrwxrwx root/root         0 2013-11-06 14:07 usr/lib64/libddicu27.so -> /opt/aster/stage/clients-odbc-linux64/DataDirect/lib/libddicu27.so
lrwxrwxrwx root/root         0 2013-11-06 10:29 usr/lib64/libiculetd.so.46 -> /opt/teradata/client/14.10/tdicu/lib64/libiculetd.so.46.0
lrwxrwxrwx root/root         0 2013-11-06 10:29 usr/lib64/libiculxtd.so.46 -> /opt/teradata/client/14.10/tdicu/lib64/libiculxtd.so.46.0

Now, as it happens you need to make sure your system is reading the related libraries provided... and most systems nowadays will use /etc/ld.so.conf.d to have simple files listing your system libraries.  We add the Aster and Teradata libraries by creating /etc/ld.so.conf.d/Zteradata.conf (named with a capital Z so it comes at the end of the default search path) with the following entries:

/opt/aster/stage/clients-odbc-linux64/Libs
/opt/aster/stage/clients-odbc-linux64/DataDirect/lib
/opt/teradata/client/14.10/odbc_64/lib
/opt/teradata/client/14.10/tdicu/lib64

The aster ODBC driver installation instructions tell you to set the DriverManagerEncoding to UTF-32, but I found if I did that it would cause a segfault, and UTF-8 managed to turn everything from Aster into Chinese on me, but fortunately UTF-16 did the trick.
Here is my /etc/aster.ini file:

[driver]
DriverManagerEncoding=UTF-16
LogLevel=0
DriverLocale=en-US
ErrorMessagesPath=/opt/aster/stage/clients-odbc-linux64/ErrorMessages
ODBCInstLib=/opt/aster/stage/clients-odbc-linux64/DataDirect/lib/libodbcinst.so

For completeness, here are my odbc.ini and odbcinst.ini files, respectively:

[ODBC]
InstallDir=/opt/teradata/client/14.10/odbc_64
Trace=0
;TraceDll=/opt/teradata/client/14.10/odbc_64/lib/odbctrac.so
;TraceFile=/usr/joe/odbcusr/trace.log
TraceAutoStop=0

[ODBC Data Sources]
TDATA=tdata.so
Aster=AsterDriver

[ASTER]
Driver=/opt/aster/stage/clients-odbc-linux64/DataDirect/lib/libAsterDriver.so
SERVER=aster.mydomain
DATABASE=beehive

[TDATA]
Driver=/opt/teradata/client/14.10/odbc_64/lib/tdata.so
Description=Teradata database
DBCName=teradata.mydomain
LastUser=
Username=
Password=
Database=
DefaultDatabase=
[ODBC DRIVERS]
Teradata=Installed
AsterDriver=Installed

[ODBC Translators]
OEM to ANSI=Installed

[AsterDriver]
Driver=/opt/aster/stage/clients-odbc-linux64/DataDirect/lib/libAsterDriver.so
IconvEncoding=UCS-4LE

[Teradata]
Driver=/opt/teradata/client/14.10/odbc_64/lib/tdata.so
APILevel=CORE
ConnectFunctions=YYY
DriverODBCVer=3.51
SQLLevel=1

Now -- after making the entries in /etc/ld.so.conf.d, and getting all the symlinks made, you have to remember to run

ldconfig -v

(well, OK, maybe you don't need -v but I always do that) to refresh ld.so's cache.
After that, voila!  Python is able to talk to both Teradata and Aster simultaneously!
Here's my test program:

#!/usr/bin/env python

import pyodbc
pyodbc.pooling = False

def Tables(conn):
    c = conn.cursor()
    tables = c.tables()
    rs = tables.fetchall()

    for r in rs:
        print "%s.%s" % (r[1], r[2])


tdata = pyodbc.connect('DSN=TDATA;UID=matt;PWD=********')
aster = pyodbc.connect('DSN=Aster;UID=matt;PWD=********')

print "Teradata tables"
Tables(tdata)

print "Aster tables"
Tables(aster)

Except of course I blanked out my password :)
Hopefully this helps both of you similarly situated users whom I've never met use both libraries simultanously! (OK, maybe there are three of you, but it isn't a big number!)
Also -- one last thing -- if you were previously using LD_LIBRARY_PATH to set the path to your libraries -- you should get rid of that, since the /etc/ld.so.conf.d/Zteradata.conf file now sets that for all users of your system.  You have to make sure the aster Libs load after the normal libc libraries because they include a rude version of the c++ runtime which will make ld cry out a warning which will annoy the heck out of you.

Forums: 

Viewing all articles
Browse latest Browse all 445

Trending Articles