2012-09-14 26 views
5

我有兩部分的問題涉及使用sp_send_dbmail將查詢結果作爲附件發送。sp_send_dbmail附加在數據庫中存儲爲varbinary的文件

問題1:只打開基本的.txt文件。任何其他格式如.pdf或.jpg都會損壞。

問題2:試圖發送多個附件時,我收到一個文件,其中所有文件名都粘在一起。

我運行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發送郵件腳本。爲簡潔起見,我排除了declare語句。

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時讀取。但顯然原始標題丟失了。值得注意的是,附件文件大小與原始文件不同。這很可能是由於不正確的編碼。所以我希望有一種方法可以使用存儲的mimetype來創建文件頭,或者以某種方式在二進制數據中包含文件頭?

我對最後幾個參數的值也沒有信心,而且我知道coalesce並不完全正確,因爲它使用逗號前綴第一個文件名。但好的文檔幾乎不可能找到。請幫忙!

回答

6

我不認爲你將能夠直接從SQL發送二進制數據。有一個few posts out there談論這個相同的問題。從Microsoft documentation返回查詢的文本被格式化爲文本文件。二進制格式爲十六進制。正如您指出的那樣,會破壞任何不是文本文檔的文件。

我認爲你仍然可以完成你正在嘗試做的事情,然而通過第一個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

在這個問題上任何運氣。 –

相關問題