为什么在Linux环境中使用pyodbc插入空值?

时间:2014-10-22 08:46:12

标签: python sql-server pyodbc freetds unixodbc

freetds的-的unixODBC-MSSQL2012

下面的conf:

/etc/odbcinst.ini

[FreeTDS]
Description = FreeTDS Driver
Driver = /usr/local/freetds/lib/libtdsodbc.so
Setup = /usr/lib64/libtdsS.so.2
FileUsage = 1
CPTimeout = 5
CRReuse = 5

/etc/odbc.ini

[MSSQLTEST]
driver=FreeTDS
server=10.10.1.16
port=1433
database=ACCOUNT
client_charset = UTF-8
tds_version = 8.0

/etc/freetds.conf

[MSSQLTEST]
host = 10.10.1.16
port = 1433
tds version = 8.0

我使用pyodbc连接MSSQL2012

import sys 
reload(sys)
sys.setdefaultencoding('utf-8')

import pyodbc
conn =  pyodbc.connect("DSN=MSSQLTEST;UID=sa;PWD=xxxxx;database=PAYON;CHARSET=UTF8")
cursor = conn.cursor()
self.cursor.execute("{call insert_name('안녕')}")

结果:MSSQL2012中的name列为空值NOT WRONG值。

self.cursor.execute("{call insert_name('123')}")

结果:MSSQL2012中的name列是123.

name是nvarchar(50)。我认为这是charset /编码问题。

Linux是cenos6.5,$ LANG是ko_KR.UTF-8。

但我不知道问题点。

其他信息

编码问题? mssql设置?

我在下面写代码:

# -*- coding:utf-8 -*-

import sys
reload(sys)
sys.setdefaultencoding('utf-8')


import pyodbc
conn = pyodbc.connect("DSN=MSSQLTEST;UID=sa;PWD=xxxxx;database=PAYON")
cursor = conn.cursor()

cursor.execute("insert into tpayon_test (name) values(N'안녕')")
cursor.commit()
conn.close()

和unixODBC日志:

[ODBC][28783][1414141316.359552][__handles.c][450]
    Exit:[SQL_SUCCESS]
        Environment = 0x1987c70
[ODBC][28783][1414141316.359682][SQLSetEnvAttr.c][182]
    Entry:            
        Environment = 0x1987c70            
        Attribute = SQL_ATTR_ODBC_VERSION            
        Value = 0x3            
        StrLen = 4
[ODBC][28783][1414141316.359740][SQLSetEnvAttr.c][349]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.359792][SQLAllocHandle.c][364]
    Entry:
        Handle Type = 2
        Input Handle = 0x1987c70
[ODBC][28783][1414141316.359865][SQLAllocHandle.c][482]
    Exit:[SQL_SUCCESS]
        Output Handle = 0x19848f0
[ODBC][28783][1414141316.359930][SQLDriverConnectW.c][286]
    Entry:            
        Connection = 0x19848f0            
        Window Hdl = (nil)            
        Str In = [DSN=MSSQLTEST;UID=sa;PWD=xxxxx;database=PAYON][length = 51]            
        Str Out = (nil)            
        Str Out Max = 0            
        Str Out Ptr = (nil)            
        Completion = 0
    UNICODE Using encoding ASCII 'ISO8859-1' and UNICODE 'UCS-2LE'

[ODBC][28783][1414141316.368017][SQLDriverConnectW.c][842]
    Exit:[SQL_SUCCESS]                    
        Connection Out [[NULL]]
[ODBC][28783][1414141316.368172][SQLSetConnectAttr.c][321]
    Entry:            
        Connection = 0x19848f0            
        Attribute = SQL_ATTR_AUTOCOMMIT            
        Value = (nil)            
        StrLen = -5
[ODBC][28783][1414141316.368872][SQLSetConnectAttr.c][675]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.368977][SQLGetInfo.c][546]
    Entry:            
        Connection = 0x19848f0            
        Info Type = SQL_DRIVER_ODBC_VER (77)            
        Info Value = 0x7fffe24b61c0            
        Buffer Length = 20            
        StrLen = 0x7fffe24b61be
[ODBC][28783][1414141316.369032][SQLGetInfo.c][608]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.369121][SQLGetInfo.c][546]
    Entry:            
        Connection = 0x19848f0            
        Info Type = SQL_DESCRIBE_PARAMETER (10002)            
        Info Value = 0x7fffe24b61a0            
        Buffer Length = 2            
        StrLen = 0x7fffe24b61be
[ODBC][28783][1414141316.369204][SQLGetInfo.c][608]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.369255][SQLGetInfo.c][546]
    Entry:            
        Connection = 0x19848f0            
        Info Type = SQL_NEED_LONG_DATA_LEN (111)            
        Info Value = 0x7fffe24b61a0            
        Buffer Length = 2            
        StrLen = 0x7fffe24b61be
[ODBC][28783][1414141316.369304][SQLGetInfo.c][608]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.369347][SQLAllocHandle.c][529]
    Entry:
        Handle Type = 3
        Input Handle = 0x19848f0
[ODBC][28783][1414141316.369411][SQLAllocHandle.c][1064]
    Exit:[SQL_SUCCESS]
        Output Handle = 0x19db6d0
[ODBC][28783][1414141316.369463][SQLGetTypeInfo.c][164]
    Entry:            
        Statement = 0x19db6d0            
        Data Type = SQL_TYPE_TIMESTAMP
[ODBC][28783][1414141316.391748][SQLGetTypeInfo.c][314]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.391810][SQLFetch.c][158]
    Entry:            
        Statement = 0x19db6d0
[ODBC][28783][1414141316.391864][SQLFetch.c][340]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.391910][SQLGetData.c][233]
    Entry:            
        Statement = 0x19db6d0            
        Column Number = 3            
        Target Type = 4 SQL_INTEGER            
        Buffer Length = 4            
        Target Value = 0x7fffe24b61b8            
        StrLen Or Ind = (nil)
[ODBC][28783][1414141316.391970][SQLGetData.c][497]
    Exit:[SQL_SUCCESS]                
        Buffer = [23]                
        Strlen Or Ind = NULLPTR
[ODBC][28783][1414141316.392015][SQLGetTypeInfo.c][164]
    Entry:            
        Statement = 0x19db6d0            
        Data Type = SQL_VARCHAR
[ODBC][28783][1414141316.392056][SQLGetTypeInfo.c][186]Error: 24000
[ODBC][28783][1414141316.392109][SQLGetTypeInfo.c][164]
    Entry:            
        Statement = 0x19db6d0            
        Data Type = Unknown(-9)
[ODBC][28783][1414141316.392196][SQLGetTypeInfo.c][186]Error: 24000
[ODBC][28783][1414141316.392275][SQLGetTypeInfo.c][164]
    Entry:            
        Statement = 0x19db6d0            
        Data Type = SQL_BINARY
[ODBC][28783][1414141316.392326][SQLGetTypeInfo.c][186]Error: 24000
[ODBC][28783][1414141316.392386][SQLFreeStmt.c][140]
    Entry:            
        Statement = 0x19db6d0            
        Option = 0
[ODBC][28783][1414141316.392542][SQLFreeStmt.c][246]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.392610][SQLAllocHandle.c][529]
    Entry:
        Handle Type = 3
        Input Handle = 0x19848f0
[ODBC][28783][1414141316.392665][SQLAllocHandle.c][1064]
    Exit:[SQL_SUCCESS]
        Output Handle = 0x1a057f0
[ODBC][28783][1414141316.392722][SQLFreeStmt.c][140]
    Entry:            
        Statement = 0x1a057f0            
        Option = 0
[ODBC][28783][1414141316.392767][SQLFreeStmt.c][246]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.392814][SQLExecDirect.c][236]
    Entry:            
        Statement = 0x1a057f0            
        SQL = [insert into tpayon_test (name) values(N'안녕')][length = 48 (SQL_NTS)]
[ODBC][28783][1414141316.394015][SQLExecDirect.c][499]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.394075][SQLRowCount.c][169]
    Entry:            
        Statement = 0x1a057f0            
        Row Count = 0x7fffe24b6300
[ODBC][28783][1414141316.394122][SQLRowCount.c][240]
    Exit:[SQL_SUCCESS]                
        Row Count = 0x7fffe24b6300 -> 1
[ODBC][28783][1414141316.394169][SQLNumResultCols.c][152]
    Entry:            
        Statement = 0x1a057f0            
        Column Count = 0x7fffe24b631e
[ODBC][28783][1414141316.394214][SQLNumResultCols.c][244]
    Exit:[SQL_SUCCESS]                
        Count = 0x7fffe24b631e -> 0
[ODBC][28783][1414141316.394265][SQLEndTran.c][318]
    Entry:            
        Connection = 0x19848f0            
        Completion Type = 0
[ODBC][28783][1414141316.394971][SQLGetInfo.c][546]
    Entry:            
        Connection = 0x19848f0            
        Info Type = SQL_CURSOR_COMMIT_BEHAVIOR (23)            
        Info Value = 0x1985d60            
        Buffer Length = 2            
        StrLen = 0x7fffe24b621e
[ODBC][28783][1414141316.395160][SQLGetInfo.c][608]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.395211][SQLGetInfo.c][546]
    Entry:            
        Connection = 0x19848f0            
        Info Type = SQL_CURSOR_ROLLBACK_BEHAVIOR (24)            
        Info Value = 0x1985d62            
        Buffer Length = 2            
        StrLen = 0x7fffe24b621e
[ODBC][28783][1414141316.395260][SQLGetInfo.c][608]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.395302][SQLEndTran.c][504]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.395349][SQLEndTran.c][318]
    Entry:            
        Connection = 0x19848f0            
        Completion Type = 1
[ODBC][28783][1414141316.408656][SQLEndTran.c][504]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.408716][SQLDisconnect.c][204]
    Entry:            
        Connection = 0x19848f0
[ODBC][28783][1414141316.408893][SQLDisconnect.c][341]
    Exit:[SQL_SUCCESS]
[ODBC][28783][1414141316.409055][SQLFreeHandle.c][279]
    Entry:
        Handle Type = 2
        Input Handle = 0x19848f0
[ODBC][28783][1414141316.409117][SQLFreeHandle.c][330]
    Exit:[SQL_SUCCESS]

2 个答案:

答案 0 :(得分:2)

我在我的Xububtu 14.04盒子上用pyodbc和FreeTDS进行了一些测试,发现如果我使用了参数化查询,我的代码工作正常,但它不适用于文字查询。也就是说,使用与您相同的FreeTDS / ODBC配置,此方法失败(插入不同的字符):

# -*- coding:utf-8 -*-
import pyodbc
conn = pyodbc.connect("DSN=MSSQLTEST;UID=sa;PWD=whatever;database=myDb")
cursor = conn.cursor()

cursor.execute(u"insert into tpayon_test (name) values (N'안녕')")

cursor.commit()
conn.close()

虽然这种方法运行良好:

# -*- coding:utf-8 -*-
import pyodbc
conn = pyodbc.connect("DSN=MSSQLTEST;UID=sa;PWD=whatever;database=myDb")
cursor = conn.cursor()

sql = "insert into tpayon_test (name) values (?)"
parameters = [u"안녕"]
cursor.execute(sql, parameters)

cursor.commit()
conn.close()

同样,这(及其变体)不起作用

cursor.execute(u"{call insert_name(N'안녕')}")

但是这样做了:

sql = "{call insert_name(?)}"
parameters = [u"안녕"]
cursor.execute(sql, parameters)

答案 1 :(得分:1)

给它一个旋转?

name_to_insert = '안녕'
name_to_insert.encode('utf-8')
self.cursor.execute("{call insert_name(N'" + name_to_insert + "')}")

希望这样做 - 或者您可以设置如上所述的标题,但我更喜欢在代码中明确。