Monday, June 24, 2019

Linked Server Refering to Self

Likely a problem with DNS tables, this will fix it with a valid SQL user.

-- EXEC sp_addlinkedserver [MyServer], '', 'SQLOLEDB', '1.2.3.4'

Friday, June 21, 2019

sp_send_dbmail solution for attachments


When using the @query parameter with sp_send_dbmail via the SQL Agent to send an attachment in the email you may get the following error:

“Failed to initialize sqlcmd library with error number -2147467259. [SQLSTATE 42000] (Error 22050).  The step failed.”

Apparently this is the sp_send_dbmail equivalent of the nearly useless Visual Basic “Object variable or With block variable not set.” error.

Note that the syntax was correct and everything worked fine when operating from a query session.

One way to get around it is to preface the execution context with a power user:

EXECUTE AS LOGIN = 'ElmerFudd'

EXEC msdb.dbo.sp_send_dbmail
@profile_name = 'Alert Stuff',
@recipients = 'bugz@bunny.org; ',
@subject = 'Favorite vacations',
@query = N'
use msdb;
go
select top 10 ...

This took quite an effort to solve so I decided to share …

Thursday, May 30, 2019

FYI: SS2016 Requires .Net FW 3.5 for DBMail

This was painful enough to figure out that I figured I would share the information with the entire group.

Instructions for installs on Server 2012+: https://docs.microsoft.com/en-us/windows-hardware/manufacture/desktop/enable-net-framework-35-by-using-the-add-roles-and-features-wizard (will not install from a exe).

 Detailz …

 DatabaseMail.exe (in the instance Binn directory) which is invoked by the Service Broker which is part of the Core database engine since 2005 requires .Net FW 3.5, likely as 3.5 came out in 2002 and is a dependency. Why it will not work with a more recent version of .Net is a mystery, perhaps they are not all backwards compatible (https://docs.microsoft.com/en-us/dotnet/framework/migration-guide/version-compatibility).

Essentially there was no error information available, mail items would just sit in the queue.

Starting the exe itself on the server or invoking dbmail via a sql interface puts this in the application log of the server: “The activated proc '[dbo].[sp_sysmail_activate]' running on queue 'msdb.dbo.ExternalMailQueue' output the following: 'Could not create DatabaseMail.exe process. Executing API 'CreateProcess' failed with error number 14001.'” ~ which, after a lot of time with The Google, led to the answer.

Don’t fall for missing DatabaseMail.exe.config threads as that is a red herring. 2¢, J++

Tools:


This was painful enough to figure out that I figured I would share the information with the entire group.



Detailz …

DatabaseMail.exe (in the instance Binn directory) which is invoked by the Service Broker which is part of the Core database engine since 2005 requires .Net FW 3.5, likely as 3.5 came out in 2002 and is a dependency. Why it will not work with a more recent version of .Net is a mystery, perhaps they are not all backwards compatible (https://docs.microsoft.com/en-us/dotnet/framework/migration-guide/version-compatibility).

Essentially there was no error information available, mail items would just sit in the queue.

Starting the exe itself on the server or invoking dbmail via a sql interface puts this in the application log of the server:

“The activated proc '[dbo].[sp_sysmail_activate]' running on queue 'msdb.dbo.ExternalMailQueue' output the following:  'Could not create DatabaseMail.exe process. Executing API 'CreateProcess' failed with error number 14001.'”
~
Which, after a lot of time with The Google, led to the answer. Don’t fall for missing DatabaseMail.exe.config threads as that is a red herring.

2¢, J++

Tools:  

SELECT
    CASE sent_status
        WHEN 0 THEN 'Unsent'
        WHEN 1 THEN 'Sent'
        WHEN 2 THEN 'Failed'
        WHEN 3 THEN 'Retrying'
        END AS sent_status_desc
FROM msdb..sysmail_mailitems

SELECT *
FROM   msdb..sysmail_event_log
ORDER BY log_id DESC

SELECT @@VERSION

SELECT sent_status, send_request_date a, *
FROM msdb.dbo.sysmail_mailitems
ORDER BY a DESC

SELECT *
FROM msdb.dbo.sysmail_sentitems

USE msdb
SELECT sent_status, *
FROM sysmail_allitems

SELECT is_broker_enabled
FROM sys.databases
WHERE [name] = 'msdb';

EXECUTE msdb.dbo.sysmail_help_status_sp

-- sp_configure 'Database Mail XPs'

SELECT *
FROM msdb.dbo.sysmail_event_log
ORDER BY log_date DESC


-- select @@VERSION

SELECT *
FROM   Msdb.dbo.sysmail_profile

SELECT *
FROM   Msdb.dbo.sysmail_event_log order by log_date desc

SELECT *
FROM   Msdb.dbo.sysmail_faileditems

EXEC Msdb.dbo.sysmail_help_queue_sp

Thursday, May 16, 2019

SCRIPTING SQL SERVER OBJECTS WITH DBATOOLS – BEYOND DEFAULT OPTIONS

https://claudioessilva.eu/2019/05/15/scripting-sql-server-objects-with-dbatools-beyond-default-options/ 

Probably you had the need to script out some objects from a SQL Server instance/database and this is quite easy. You just need to right click on the object (well…not on every single one, try it with an Availability Group :-), no script option available) select “Script XXXX as” and you have it. But have you realized that this option doesn’t bring all the stuff? Let’s say you are scripting a table and you have a Non-Clustered index or a trigger…using this option some of the objects under the table will not be scripted out. 

I understand this as each one of this is a different object, which means if you go to each one you can right click and script just that one. SSMS – “Generate Scripts…” option. This is a tedious work and you can easily miss some objects. What can we do on SSMS to make this task easier? You can accomplish this task by using the “Generate Scripts…” option under “Tasks” when you right-click on the database: This will open the wizard and at the “Set Scripting Options” you can just click on the “Advanced” button and there you can change the properties. Here you can see that some defaults are in place, and “Script Indexes” is one of them. This is much easier right? All-in-one in a single scripting operation. What about automating this task? You want to script out multiple objects from different instances/databases. Enter dbatools’ “Export-” commands To search for commands within dbatools we can use the Find-DbaCommand. 1 Find-DbaCommand -Tag Export This command give to us a nice Synopsis text that help to find which command is the one we want. From the output we can see that we have (as of v0.9.824) 5 commands tagged as Export. To replicate our SSMS example using PowerShell we will use the Export-DbaScript. Don’t forget to use Get-Help cmdlet to find the available parameters and get some examples on how you can use the command. 1 Get-Help Export-DbaScript -Detailed Tip: Try other switches like -Examples or even -ShowWindow (won’t work on PSCore but dbatools does!) switches. Example using our “MyTable” object Here is how MyTable looks like: 3 Columns 1 Default constraint 1 Non-Clustered Index 1 Get-DbaDbTable -SqlInstance SQL1 -Database DB1 -Table MyTable | Export-DbaScript -Passthru Note: I’m using -PassThru parameter to output the script to the console, by default it will create a SQL file. The output of this execution is even more incomplete when comparing with SSMS. Here, we dont even get the default constraint scripted. Using “New-DbaScriptingOption” command dbatools has a command that makes it possible to create an object of type ScriptingOptions. Then we can change the properties like we have done before on the “Generate Scripts…” option on SSMS. 1 2 $options = New-DbaScriptingOption $options | Get-Member Use Get-Member so you can see what properties the object offers. Here we start seeing what we need. By default what are the values of properties like NonClusteredIndexes and DriDefaults 1 2 3 $options = New-DbaScriptingOption $options.NonClusteredIndexes $options.DriDefaults False! That explains why they are “missing” from our default Export-DbaScript output. NOTE: Export-DbaUser comamnd can also leverage on this object. Try it. Let’s change this options to $true and pass our $options object as the value of the -ScriptingOptionsObject parameter and run the command again. 1 2 3 4 $options = New-DbaScriptingOption $options.NonClusteredIndexes = $true $options.DriDefaults = $true Get-DbaDbTable -SqlInstance SQL1 -Database DB1 -Table MyTable | Export-DbaScript -Passthru -ScriptingOptionsObject $options Nice! Now we can see all the stuff. Try it yourself See the other options available, change the values, rerun and analyse the output. Do you need to export it to run on a lower SQL Server version? Change the TargetServerVersion option 1 2 #Will script the code like SQL Server 2014 $options.TargetServerVersion = "Version120" You want to include the “IF NOT EXISTS” statement? Change the “IncludeIfNotExists” option. Here is an example to script out a list of tables (add more to the $TableName variable): 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 $SourceServer = "SQL1"; $SourceDB = "DB1"; $TableName = "MyTable", 'YourTable'; $options = New-DbaScriptingOption $options.DriPrimaryKey = $true $options.DriForeignKeys = $true $options.DriNonClustered = $true $options.DriChecks = $true $options.DriDefaults = $true $options.Indexes = $true $options.IncludeIfNotExists = $true $TableName | Foreach-Object { Get-DbaDbTable -ServerInstance $SourceServer -Database $SourceDB -Table $_ | Export-DbaScript -ScriptingOptionsObject $options -Passthru; } Availability Groups example using dbatools Try it yourself: 1 Get-DbaAvailabilityGroup -SqlInstance SQL1 -AvailabilityGroup SQL1_AG1 | Export-DbaScript -Passthru Summary We have seen how we can leverage on some dbatools commands to generate T-SQL scripts from objects. This way we can versioning or just run them on other instace/database. We have seen what default options it (doesn’t) brings and how to add more options to it. We also saw that, for some objects, the script option is not available. As example Availability Groups, but dbatools can help you here too. Thanks for reading!