Преобразование строковой переменной в GUID
Я хочу использовать Execute SQL Task
запрос возвращает varbinary
данные из таблицы базы данных. Запрос требует UniqueIdentifier
как параметр хранящийся в string
в переменной пакета. Результат этого запроса (данные varbinary) будет сохранен во второй переменной типа Byte.
в таблице ниже показан мой список локальных переменных, обратите внимание, что DocumentGuid предварительно заполнен другой частью потока управления
| Variable Name | Qualified Name | Scope | Data Type | Comments |
|---------------|--------------------|---------|-----------|------------------------------------------|
| DocumentGuid | User::DocumentGuid | Package | String | Used to store a GUID value |
| DocumentData | User::DocumentData | Package | Byte | Used to hold varbinary data from a query |
когда я пытаюсь использовать это в а Execute SQL Task
запрос наподобие этого:
SELECT DocData
FROM docsRepo
WHERE DocumentGuid = ?
передача параметров как
| Variable name | Direction | Data Type | Parameter Name | Parameter Size |
|--------------------|-----------|-----------|----------------|----------------|
| User::DocumentGuid | Input | GUID | 0 | 100 |
И Результирующий Набор
| Result Name | Variable Name |
|-------------|--------------------|
| DocData | User::DocumentData |
я получаю следующую ошибку:
[выполнить задачу SQL] ошибка: выполнение запроса " выберите DocData из
dbo.доксрепо..."ошибка со следующей ошибкой:" ошибка преобразования при преобразовании из символьной строки в uniqueidentifier.". Возможные причины сбоя: проблемы с запросом, " ResultSet" свойство задано неправильно, параметры заданы неправильно, или соединение установлено неправильно.
мне не хватает какой-то фундаментальной логики здесь? Следующий скрипт отлично работает в SQL Server:
SELECT DocData
FROM docsRepo
WHERE DocumentGuid = '53A394A7-5D2B-40C0-A04D-90553E4535C3'
4 ответов
я внес два изменения в ваш образец, и я смог заставить его работать.
первым было изменение типа параметра, который будет строкой в задаче Execute SQL. Я знаю, это guid, но поскольку это строковая переменная в SSIS, сохраните выравнивание типа данных и позвольте серверной СУБД/драйверу обрабатывать преобразование.
другая вещь, которую я изменил, - это тип данных для DocData. Вы определили как Byte, но никогда не использовали это, я беспокоюсь, что это один целый байт, а не байты. Во всяком случае, для таблицы, которую я создал, использование типа данных объекта для служб SSIS заставило его работать.
источник запроса
вот моя таблица и одно значение в ней
IF NOT EXISTS
(
SELECT
*
FROM
sys.tables
WHERE
name = 'docsRepo'
)
BEGIN
CREATE TABLE dbo.docsRepo
(
DocumentGuid uniqueidentifier
, DocumentData varbinary(MAX)
);
INSERT INTO
dbo.docsRepo
SELECT
'53A394A7-5D2B-40C0-A04D-90553E4535C3'
, CAST('Hello world' AS varbinary(MAX));
END;
- поисковых запросов
SELECT D.DocumentData FROM dbo.docsRepo AS D WHERE D.DocumentGuid = ?;
настроен с полным набором результатов. Используется драйвер OLE DB. Имя параметра 0, Тип данных varchar, переменная User:: DocumentGuid. Вкладка "результаты", у меня есть имя результирующего набора 0, переменная User:: DocumentData
этой Мбзм создаст SSIS пакет, который демонстрирует все это
<Biml xmlns="http://schemas.varigence.com/biml.xsd">
<Connections>
<OleDbConnection Name="localhost" ConnectionString="Provider=SQLNCLI11;Integrated Security=SSPI;Data Source=.\dev2014;Initial Catalog=tempdb" />
</Connections>
<Packages>
<Package Name="SO_50028154" ConstraintMode="Linear">
<Variables>
<Variable Name="DocumentGuid" DataType="String">53A394A7-5D2B-40C0-A04D-90553E4535C3</Variable>
<Variable Name="DocumentData" DataType="Object" />
</Variables>
<Tasks>
<ExecuteSQL Name="SQL GenerateTable" ConnectionName="localhost">
<DirectInput>IF NOT EXISTS(SELECT * FROM sys.tables WHERE name='docsRepo')BEGIN CREATE TABLE dbo.docsRepo(DocumentGuid uniqueidentifier, DocumentData varbinary(max)); INSERT INTO dbo.docsRepo SELECT '53A394A7-5D2B-40C0-A04D-90553E4535C3', CAST('Hello world' AS varbinary(max)); END</DirectInput>
</ExecuteSQL>
<ExecuteSQL Name="SQL Retrieve data" ConnectionName="localhost" ResultSet="Full">
<DirectInput>SELECT D.DocumentData FROM dbo.docsRepo AS D WHERE D.DocumentGuid = ?;</DirectInput>
<Parameters>
<Parameter DataType="AnsiString" Name="0" VariableName="User.DocumentGuid" />
</Parameters>
<Results>
<Result Name="0" VariableName="User.DocumentData" />
</Results>
</ExecuteSQL>
</Tasks>
</Package>
</Packages>
</Biml>
вы можете попробовать explicit CAST
:
SELECT DocData
FROM docsRepo
WHERE DocumentGuid = CAST(? AS UNIQUEIDENTIFIER);
похоже, что у вас могут быть значения, хранящиеся в DocumentGuid
которые не являются допустимыми GUID или недопустимыми GUID, которые вы передаете в запрос
чтобы найти, содержит ли ваша таблица недопустимые GUID, вы можете запустить:
SELECT DocumentGuid
FROM docsRepo
WHERE TRY_CAST(DocumentGuid AS UNIQUEIDENTIFIER) IS NULL
Если вы хотите отправить свой uniqueIdentifer как varchar в задачу Execute SQL, вы можете использовать этот запрос в своей задаче Execute SQL:
DECLARE @docGuid AS UniqueIdentifier
SET @docGuid = CAST ( '{' + ? + '}' AS UniqueIdentifier )
SELECT DocData
FROM docsRepo
WHERE DocumentGuid = @docGuid
the ? отправить по параметру как varchar с размером 50