Apache Ignite 2.7: ODBC Linux - возвращены неверные данные - PullRequest
2 голосов
/ 27 апреля 2019

Я тестирую драйвер Apache Ignite 2.7 для Linux ODBC, и по какой-то причине он работает неправильно.

Когда я запускаю инструкцию SQL, например SELECT Id FROM MyTable, значение «искажается», возвращаязначение типа 8\00\05\07\09\0d\09\08\0-\0 для столбца Id.Это похоже на ожидаемое мной значение, равное 80579d98-9010-4610-b12e-ed33ed7d3c62.Я не знаю, являются ли они нулевыми символами ASCII (\0), или как, но что-то не так, и я не могу понять это.

enter image description here

Когда я использую инструмент SQLLine, значения не "искажаются".Я также установил драйвер ODBC для Windows, и он работал нормально, но мне нужен драйвер ODBC для Linux.

enter image description here

Я создал драйвер со следующими библиотеками:

gcc 6.3.0 20170516 (Debian 6.3.0-18+deb9u1)
GNU Make 4.1
libtoolize 2.4.6
aclocal 1.15
autoheader 2.69
automake 1.15
autoreconf 2.69
unixodbc 2.3.4-1
libssl 1.0.2r-1~deb9u1

Вот содержимое файла ignite-odbc-install.ini:

# cat /src/apache-ignite-2.7.0-bin/platforms/cpp/odbc/install/ignite-odbc-install.ini
[Apache Ignite]
Description=Apache Ignite
Driver=/usr/local/lib/libignite-odbc.so
Setup=/usr/local/lib/libignite-odbc.so
DriverODBCVer=03.00
FileUsage=0

Вот результат запуска odbcinst:

# odbcinst -i -d -f /src/apache-ignite-2.7.0-bin/platforms/cpp/odbc/install/ignite-odbc-install.ini
odbcinst: Driver installed. Usage count increased to 1.
    Target directory is /etc

Вотсодержимое папки / usr / local / lib:

# cd /usr/local/lib
# ls
libignite-binary-2.7.0.33575.so.0      libignite-common-2.7.0.33575.so.0      libignite-odbc-2.7.0.33575.so.0      libignite-thin-client-2.7.0.33575.so.0
libignite-binary-2.7.0.33575.so.0.0.0  libignite-common-2.7.0.33575.so.0.0.0  libignite-odbc-2.7.0.33575.so.0.0.0  libignite-thin-client-2.7.0.33575.so.0.0.0
libignite-binary.a                     libignite-common.a                     libignite-odbc.a                     libignite-thin-client.a
libignite-binary.la                    libignite-common.la                    libignite-odbc.la                    libignite-thin-client.la
libignite-binary.so                    libignite-common.so                    libignite-odbc.so                    libignite-thin-client.so

Вот результат команды ldd:

# ldd /usr/local/lib/libignite-odbc.so
        linux-vdso.so.1 (0x00007fffa46d0000)
        libdl.so.2 => /lib/x86_64-linux-gnu/libdl.so.2 (0x00007f18d554c000)
        libignite-binary-2.7.0.33575.so.0 => /usr/local/lib/libignite-binary-2.7.0.33575.so.0 (0x00007f18d531d000)
        libodbcinst.so.2 => /usr/lib/x86_64-linux-gnu/libodbcinst.so.2 (0x00007f18d5108000)
        libstdc++.so.6 => /usr/lib/x86_64-linux-gnu/libstdc++.so.6 (0x00007f18d4d86000)
        libm.so.6 => /lib/x86_64-linux-gnu/libm.so.6 (0x00007f18d4a82000)
        libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f18d46e3000)
        libgcc_s.so.1 => /lib/x86_64-linux-gnu/libgcc_s.so.1 (0x00007f18d44cc000)
        /lib64/ld-linux-x86-64.so.2 (0x00007f18d59f8000)
        libignite-common-2.7.0.33575.so.0 => /usr/local/lib/libignite-common-2.7.0.33575.so.0 (0x00007f18d42b5000)
        libltdl.so.7 => /usr/lib/x86_64-linux-gnu/libltdl.so.7 (0x00007f18d40ab000)
        libpthread.so.0 => /lib/x86_64-linux-gnu/libpthread.so.0 (0x00007f18d3e8e000)

UPDATE 1

Я запустил это изКонсольная программа .Net Core 2.2 C #, содержащая только ссылку NuGet System.Data.Odbc 4.5.0 в среде Docker Linux.

IgniteGarbledDataReproducer.csproj

<Project Sdk="Microsoft.NET.Sdk">

  <PropertyGroup>
    <OutputType>Exe</OutputType>
    <TargetFramework>netcoreapp2.2</TargetFramework>
  </PropertyGroup>

  <ItemGroup>
    <PackageReference Include="System.Data.Odbc" Version="4.5.0" />
  </ItemGroup>

</Project>

Program.cs

using System;
using System.Data.Odbc;

namespace IgniteGarbledDataReproducer {
    internal class Program {
        private static void Main(string[] args) {
            Console.WriteLine("START");

            var connectionString = Environment.GetEnvironmentVariable("IGNITE_CONNECTION_STRING");
            Console.WriteLine(connectionString);

            using (var conn = new OdbcConnection(connectionString)) {
                conn.Open();

                Console.WriteLine("Connection Opened");

                using (var cmd = conn.CreateCommand()) {
                    cmd.CommandText = "DROP TABLE IF EXISTS MyTable;";
                    cmd.ExecuteNonQuery();
                }

                Console.WriteLine("Table Dropped (if it existed)");

                using (var cmd = conn.CreateCommand()) {
                    cmd.CommandText = "CREATE TABLE MyTable (Id varchar(36) not null, FirstName varchar(255), LastName varchar(255), PRIMARY KEY (Id));";
                    cmd.ExecuteNonQuery();
                }

                Console.WriteLine("Table Created");

                using (var cmd = conn.CreateCommand()) {
                    cmd.CommandText = "INSERT INTO MyTable (Id) VALUES ('80579d98-9010-4610-b12e-ed33ed7d3c62');";
                    cmd.ExecuteNonQuery();
                }

                Console.WriteLine("Data Inserted");

                using (var cmd = conn.CreateCommand()) {
                    cmd.CommandText = "SELECT Id FROM MyTable;";
                    using (var rdr = cmd.ExecuteReader()) {
                        while (rdr.Read())
                            Console.WriteLine($"Id = '{rdr["Id"]}'");
                    }
                }

                Console.WriteLine("Data Returned");
            }

            Console.WriteLine("DONE");
        }
    }
}

Dockerfile

FROM microsoft/dotnet:2.2-runtime AS base
WORKDIR /app

WORKDIR /src
# Install pre-requisites for building and installing Ignite ODBC Driver
RUN apt-get update \
    && apt-get install -y gcc g++ make automake autotools-dev libtool m4 libssl1.0-dev unixodbc-dev unixodbc-bin unzip \
    && apt-get clean -y

# Download and unzip Ignite binary release
#ADD http://apache.mirrors.hoobly.com/ignite/2.7.0/apache-ignite-2.7.0-bin.zip /src/
COPY apache-ignite-2.7.0-bin.zip .
RUN unzip -q /src/apache-ignite-2.7.0-bin.zip \
    && rm /src/apache-ignite-2.7.0-bin.zip

# Build Ignite ODBC Driver
ARG IGNITE_HOME=/src/apache-ignite-2.7.0-bin
WORKDIR /src/apache-ignite-2.7.0-bin/platforms/cpp
RUN libtoolize \
    && aclocal \
    && autoheader \
    && automake --add-missing \
    && autoreconf \
    && ./configure --enable-odbc --disable-core --disable-node \
    && make \
    && make install

# Install Ignite ODBC Driver
RUN apt-get update \
    && apt-get install -y unixodbc \
    && apt-get clean -y \
    && odbcinst -i -d -f /src/apache-ignite-2.7.0-bin/platforms/cpp/odbc/install/ignite-odbc-install.ini

FROM microsoft/dotnet:2.2-sdk AS build
WORKDIR /src
COPY IgniteGarbledDataReproducer/IgniteGarbledDataReproducer.csproj IgniteGarbledDataReproducer/
RUN dotnet restore IgniteGarbledDataReproducer/IgniteGarbledDataReproducer.csproj
COPY . .
WORKDIR /src/IgniteGarbledDataReproducer
RUN dotnet build IgniteGarbledDataReproducer.csproj -c Release -o /app

FROM build AS publish
RUN dotnet publish IgniteGarbledDataReproducer.csproj -c Release -o /app

FROM base AS final
WORKDIR /app
COPY --from=publish /app .
ENTRYPOINT ["dotnet", "IgniteGarbledDataReproducer.dll"]

docker-compose.yml

version: '3.4'

services:
  ignitegarbleddatareproducer:
    image: ${DOCKER_REGISTRY-}ignite-garbled-data-reproducer
    build:
      context: .
      dockerfile: IgniteGarbledDataReproducer/Dockerfile

docker-compose-override.yml

version: '3.4'

services:
  ignitegarbleddatareproducer:
    environment:
      - ASPNETCORE_ENVIRONMENT=Development
      - IGNITE_CONNECTION_STRING=DRIVER={Apache Ignite};ADDRESS=my-ignite-3e8b8cb9a4e192af.elb.us-east-1.amazonaws.com:10800;SSL_MODE=disable;

Сборка

docker build . -t ignitegarbleddatareproducer -f IgniteGarbledDataReproducer/Dockerfile

Выполнение

PS C:\IgniteGarbledDataReproducer> docker-compose up
Starting ignitegarbleddatareproducer_ignitegarbleddatareproducer_1 ... done
Attaching to ignitegarbleddatareproducer_ignitegarbleddatareproducer_1
ignitegarbleddatareproducer_1  | START
ignitegarbleddatareproducer_1  | DRIVER={Apache Ignite};ADDRESS=my-ignite-3e8b8cb9a4e192af.elb.us-east-1.amazonaws.com:10800;SSL_MODE=disable;
ignitegarbleddatareproducer_1  | Connection Opened
ignitegarbleddatareproducer_1  | DROP TABLE IF EXISTS MyTable;
ignitegarbleddatareproducer_1  | Table Dropped (if it existed)
ignitegarbleddatareproducer_1  | CREATE TABLE MyTable (Id varchar(36) not null, FirstName varchar(255), LastName varchar(255), PRIMARY KEY (Id));
ignitegarbleddatareproducer_1  | Table Created
ignitegarbleddatareproducer_1  | INSERT INTO MyTable (Id) VALUES ('80579d98-9010-4610-b12e-ed33ed7d3c62');
ignitegarbleddatareproducer_1  | Data Inserted
ignitegarbleddatareproducer_1  | SELECT Id FROM MyTable;
ignitegarbleddatareproducer_1  | Id = '8 0 5 7 9 d 9 8 - '
ignitegarbleddatareproducer_1  | Data Returned
ignitegarbleddatareproducer_1  | DONE
ignitegarbleddatareproducer_ignitegarbleddatareproducer_1 exited with code 0

ОБНОВЛЕНИЕ 2

Я повернулся при ведении журнала ODBC путем добавления следующего к /src/apache-ignite-2.7.0-bin/platforms/cpp/odbc/install/ignite-odbc-install.ini, и вы можете получить журналы здесь .

[ODBC]
Trace=yes
TraceFile=/logs/odbc.log

Конец журнала после SELECT Id From MyTable;Для удобства ниже приводится заявление.

[ODBC][1][1556564832.827181][SQLExecDirectW.c][177]
        Entry:
            Statement = 0xf27a80
            SQL = [SELECT Id FROM MyTable;][length = 23 (SQL_NTS)]
[ODBC][1][1556564832.847882][SQLExecDirectW.c][445]
        Exit:[SQL_SUCCESS]
[ODBC][1][1556564832.852889][SQLRowCount.c][173]
        Entry:
            Statement = 0xf27a80
            Row Count = 0x7ffd7da98f00
[ODBC][1][1556564832.856831][SQLRowCount.c][247]
        Exit:[SQL_SUCCESS]
            Row Count = 0x7ffd7da98f00 -> 0
[ODBC][1][1556564832.860335][SQLNumResultCols.c][156]
        Entry:
            Statement = 0xf27a80
            Column Count = 0x7ffd7da98f50
[ODBC][1][1556564832.864944][SQLNumResultCols.c][251]
        Exit:[SQL_SUCCESS]
            Count = 0x7ffd7da98f50 -> 1
[ODBC][1][1556564832.869909][SQLFetch.c][162]
        Entry:
            Statement = 0xf27a80
[ODBC][1][1556564832.884876][SQLFetch.c][351]
        Exit:[SQL_SUCCESS]
[ODBC][1][1556564832.894173][SQLColAttributeW.c][156]
        Entry:
            Statement = 0xf27a80
            Column Number = 1
            Field Identifier = SQL_DESC_NAME
            Character Attr = 0xf2e190
            Buffer Length = 4096
            String Length = 0x7ffd7da98f20
            Numeric Attribute = 0x7ffd7da98ee0
[ODBC][1][1556564832.898409][SQLColAttributeW.c][523]
        Exit:[SQL_SUCCESS]
[ODBC][1][1556564832.910800][SQLColAttributeW.c][156]
        Entry:
            Statement = 0xf27a80
            Column Number = 1
            Field Identifier = SQL_DESC_CONCISE_TYPE
            Character Attr = 0xf2e190
            Buffer Length = 4096
            String Length = 0x7ffd7da98f40
            Numeric Attribute = 0x7ffd7da98f00
[ODBC][1][1556564832.914551][SQLColAttributeW.c][523]
        Exit:[SQL_SUCCESS]
[ODBC][1][1556564832.925913][SQLGetData.c][237]
        Entry:
            Statement = 0xf27a80
            Column Number = 1
            Target Type = -8 SQL_WCHAR
            Buffer Length = 4094
            Target Value = 0xf2e190
            StrLen Or Ind = 0x7ffd7da98ee0
[ODBC][1][1556564832.929959][SQLGetData.c][534]
        Exit:[SQL_SUCCESS]                
            Buffer = [8](unicode)                
            Strlen Or Ind = 0x7ffd7da98ee0 -> 36
[ODBC][1][1556564832.934835][SQLFetch.c][162]
        Entry:
            Statement = 0xf27a80
[ODBC][1][1556564832.938930][SQLFetch.c][351]
        Exit:[SQL_NO_DATA]
[ODBC][1][1556564832.943970][SQLMoreResults.c][162]
        Entry:
            Statement = 0xf27a80
[ODBC][1][1556564832.960311][SQLMoreResults.c][345]
        Exit:[SQL_NO_DATA]
[ODBC][1][1556564832.965960][SQLFreeStmt.c][144]
        Entry:
            Statement = 0xf27a80
            Option = 0
[ODBC][1][1556564832.970025][SQLFreeStmt.c][266]
        Exit:[SQL_SUCCESS]
[ODBC][1][1556564832.975135][SQLFreeHandle.c][381]
        Entry:
            Handle Type = 3
            Input Handle = 0xf27a80
[ODBC][1][1556564832.981240][SQLFreeHandle.c][494]
        Exit:[SQL_SUCCESS]
[ODBC][1][1556564832.991337][SQLDisconnect.c][208]
        Entry:
            Connection = 0xf0a6f0
[ODBC][1][1556564832.996778][SQLDisconnect.c][379]
        Exit:[SQL_SUCCESS]
[ODBC][1][1556564833.001505][SQLFreeHandle.c][284]
        Entry:
            Handle Type = 2
            Input Handle = 0xf0a6f0
[ODBC][1][1556564833.006038][SQLFreeHandle.c][333]
        Exit:[SQL_SUCCESS]

ОБНОВЛЕНИЕ 3

У меня есть догадка, что это может быть связано с UTF-16 из-за аналогичного вопроса , гдесимволы разделяются \0.

Может быть, что-то делать с настройкой UNICODE в этой строке из журналов ODBC?

[ODBC][1][1556564832.214691][SQLDriverConnectW.c][290]
        Entry:
            Connection = 0xf0a6f0
            Window Hdl = (nil)
            Str In = [DRIVER={Apache Ignite};ADDRESS=my-ignite-3e8b8cb9a4e192af.elb.us-east-1.amazonaws.com:10800;SSL_MODE=disable;][length = 115 (SQL_NTS)]
            Str Out = (nil)
            Str Out Max = 0
            Str Out Ptr = 0x7ffd7da98dd0
            Completion = 0
        UNICODE Using encoding ASCII 'ANSI_X3.4-1968' and UNICODE 'UCS-2LE'

ОБНОВЛЕНИЕ 4

Я нашелэто документ от Microsoft , в котором указано

Если кодировка символов ASCII не UTF-8, например:

UNICODE Using encoding ASCII 'ISO8859-1' and UNICODE 'UCS-2LE' Существует более одного драйвераДиспетчер установлен, и ваше приложение использует неправильный, или Диспетчер драйверов не был построен правильно.

Я предполагаю, что мой драйвер собран не правильно.

И этот документ сказал использовать --enable-iconv --with-iconv-char-enc=UTF8 --with-iconv-ucode-enc=UTF16LE.

ОБНОВЛЕНИЕ 5

Я также нашел это проблема git , связанная с .Net Core, unixODBC и UTF16.

ОБНОВЛЕНИЕ 6

Я пытался добавить --enable-iconv --with-iconv-char-enc=UTF8 --with-iconv-ucode-enc=UTF16LE при вызове unixODBC configure, что изменило соединение ODBC на UNICODE Using encoding ASCII 'UTF8' and UNICODE 'UTF16LE', но я получил те же плохие данные обратно независимо от этого.Я также попробовал --with-iconv-ucode-enc=UNICODE, но без изменений.

Так я изменил свой Dockerfile.Я удалил unixodbc из apt-get и добавил в свой Dockerfile следующее

ADD ftp://ftp.unixodbc.org/pub/unixODBC/unixODBC-2.3.7.tar.gz .
RUN gunzip unixODBC*.tar.gz && tar xvf unixODBC*.tar

WORKDIR /src/unixODBC-2.3.7
RUN ./configure --enable-gui=no --enable-drivers=no --enable-iconv --with-iconv-char-enc=UTF8 --with-iconv-ucode-enc=UTF16LE
RUN make && make install

ENV LD_LIBRARY_PATH="/usr/local/lib"

И вот как теперь выглядит журнал odbc:

[ODBC][7][1556723781.839804][SQLDriverConnectW.c][290]
        Entry:
            Connection = 0x1ddff70
            Window Hdl = (nil)
            Str In = [DRIVER={Apache Ignite};ADDRESS=my-ignite-3e8b8cb9a4e192af.elb.us-east-1.amazonaws.com:10800;SSL_MODE=disable;][length = 115 (SQL_NTS)]
            Str Out = (nil)
            Str Out Max = 0
            Str Out Ptr = 0x7ffeca3331a0
            Completion = 0
        UNICODE Using encoding ASCII 'UTF8' and UNICODE 'UTF16LE'

UPDATE 7

Я подключился через isql и получил правильный ответ, но когда я подключился через iusql, я получил "плохие данные" от моего репродуктора.

ОБНОВЛЕНИЕ 8

Добавление некоторыхИнформация об установке odbc:

root@4efdb5ed98b5:/app# odbc_config --cflags
-DHAVE_UNISTD_H -DHAVE_PWD_H -DHAVE_SYS_TYPES_H -DHAVE_LONG_LONG -DSIZEOF_LONG_INT=8 -I/usr/local/include
root@4efdb5ed98b5:/app# odbc_config --ulen
-DSIZEOF_SQLULEN=8
root@4efdb5ed98b5:/app# odbcinst -j
unixODBC 2.3.7
DRIVERS............: /usr/local/etc/odbcinst.ini
SYSTEM DATA SOURCES: /usr/local/etc/odbc.ini
FILE DATA SOURCES..: /usr/local/etc/ODBCDataSources
USER DATA SOURCES..: /root/.odbc.ini
SQLULEN Size.......: 8
SQLLEN Size........: 8
SQLSETPOSIROW Size.: 8

1 Ответ

0 голосов
/ 28 мая 2019

Это ошибка в Apache Ignite 2.7.0, поэтому ответа нет. Проблема была подана с Apache Ignite, чтобы исправить это.

http://issues.apache.org/jira/browse/IGNITE-11845

...