2012-09-14 14 views
5

sp_send_dbmailを使用してクエリ結果を添付ファイルとして送信することに関する2つの質問があります。sp_send_dbmailデータベースにvarbinaryとして保存されているファイルを添付します。

問題1:基本的な.txtファイルのみが開きます。 .pdfや.jpgなどの他の形式は壊れています。

問題2:複数の添付ファイルを送信しようとすると、すべてのファイル名が貼り付けられたファイルが1つ受信されます。

私は、SQL Server 2005を実行していると私は、アップロードされたドキュメントを格納するテーブルを持っている:

CREATE TABLE [dbo].[EmailAttachment](
[EmailAttachmentID] [int] IDENTITY(1,1) NOT NULL, 
[MassEmailID] [int] NULL, -- foreign key 
[FileData] [varbinary](max) NOT NULL, 
[FileName] [varchar](100) NOT NULL, 
[MimeType] [varchar](100) NOT NULL 

私も標準の電子メールのものでMassEmailテーブルを持っています。ここに、SQL Send Mailスクリプトがあります。簡潔にするため、宣言文は除外しました。

while ((select count(MassEmailID) from MassEmail where status = 20)>0) 
begin 
    select @MassEmailID = Min(MassEmailID) from MassEmail where status = 20 
    select @Subject = [Subject] from MassEmail where MassEmailID = @MassEmailID 
    select @Body = Body from MassEmail where MassEmailID = @MassEmailID 

    set @query = 'set nocount on; select cast(FileData as varchar(max)) from Mydatabase.dbo.EmailAttachment where MassEmailID = '+ CAST(@MassEmailID as varchar(100)) 

    select @filename = '' 
    select @filename = COALESCE(@filename+ ',', '') +FileName from EmailAttachment where MassEmailID = @MassEmailID 

exec msdb.dbo.sp_send_dbmail  
    @profile_name = 'MASS_EMAIL', 
    @recipients = '[email protected]', 
    @subject = @Subject, 
    @body [email protected], 
    @body_format ='HTML', 
    @query = @query, 
    @query_attachment_filename = @filename, 
    @attach_query_result_as_file = 1, 
    @query_result_separator = '; ', 
    @query_no_truncate = 1, 
    @query_result_header = 0; 

update MassEmailset status= 30,SendDate = GetDate() where MassEmailID = @MassEmailID 
end 

バイナリデータが破損していないことを知っているので、データベースからファイルを正常に読み取ることができました。

.txtファイルは、FilaDataをvarcharにキャストしたときにのみ読み込まれます。しかし、明らかに元のヘッダーは失われています。添付ファイルのサイズが元のファイルと異なることにも注意してください。これは、不適切なエンコーディングも原因である可能性が最も高いです。ですから、格納されているMIMEタイプを使ってファイルヘッダを作成する方法や、ファイルヘッダをバイナリデータに含める方法があると思いますか?

また、私は最後のいくつかのパラメータの値に自信がありません。また、最初のファイル名の前にコンマを付けるので、合体が正しくないことが分かります。しかし、良い文書はほとんど見つけることが不可能です。助けてください!

答えて

6

SQLからバイナリデータを直接送信することはできません。同じ問題について話しているfew posts out thereがあります。 Microsoft documentationから、テキストへのテキストの戻りはテキストファイルとしてフォーマットされます。バイナリは16進数でフォーマットされます。あなたが指摘したように、それはテキスト文書ではないファイルを壊滅させます。

しかし、最初にusing BCPを実行してバイナリデータをファイルシステムにエクスポートしてから、sendmailで利用可能な従来のファイル添付方法でインポートすることで、やっていることを達成できると思います。

このようなものです。 (概念のみ - 未テストコード)

DECLARE @OutputFileAndPath VarChar(500) = '\\Log_Files\MyFile.pdf ' 
DECLARE @sql VarChar(8000) 

SELECT @sql = 'BCP "SELECT MyFile FROM [dbo].[MyTable] 
    WHERE PrimaryKey = 12345" queryout ' + @OutputFileAndPath + 
     ' -S MyServer\MyInstance -T -fC:\Documents.fmt' 

/* you could use a generic format file that would cover most formats */ 

EXEC xp_cmdshell @sql, NO_OUTPUT; 

while ((select count(MassEmailID) from MassEmail where status = 20)>0) 
begin 
    select @MassEmailID = Min(MassEmailID) from MassEmail where status = 20 
    select @Subject = [Subject] from MassEmail where MassEmailID = @MassEmailID 
    select @Body = Body from MassEmail where MassEmailID = @MassEmailID 


    exec msdb.dbo.sp_send_dbmail  
     @profile_name = 'MASS_EMAIL', 
     @recipients = '[email protected]', 
     @subject = @Subject, 
     @body [email protected], 
     @body_format ='HTML', 
     @file_attachments = @OutputFileAndPath /* i.e. \\Log_Files\MyFile.pdf */ 

    update MassEmailset status= 30,SendDate = GetDate() where MassEmailID = @MassEmailID 
end  
+0

ご回答ありがとうございます。動作する可能性がありますが、特にユーザーがアップロードしたファイルでは、セキュリティ上の制限によりxp_cmdshellの実行が禁止されています。私は別のアプローチを取らなければならないように見えます。 –

+0

この問題は不運です。 –

関連する問題