Wednesday, October 7, 2009

Script to identify unused indexes

While indexes can increase the read performance, they can decrease the write performance during any updates. Any time a query modifies the data in a table the indexes on the data also must change. Any database that has a large number of transactions modifying the data will perform good with fewer indexes. But the key is striking a balance between fewer indexes that can improve the write performance and more indexes to improve the read performance.

Below is a query that can give useful statistics of how the indexes are being used. If the number of seeks and number of scans is 0 with the number of updates significantly large, then there is no point in having the index. The below is a good starting point for assessing and distinguishing between useful and unused indexes.



USE DBNAME
GO
SELECT DB_NAME(DATABASE_ID) AS [DATABASE NAME]
 , OBJECT_NAME(SS.OBJECT_ID) AS [OBJECT NAME]
 , I.NAME AS [INDEX NAME]
-- , I.INDEX_ID AS [INDEX ID]
 , USER_SEEKS AS [NUMBER OF SEEKS]
 , USER_SCANS AS [NUMBER OF SCANS]
 , USER_LOOKUPS AS [NUMBER OF BOOKMARK LOOKUPS]
 , USER_UPDATES AS [NUMBER OF UPDATES]
FROM   
   SYS.DM_DB_INDEX_USAGE_STATS SS
   INNER JOIN SYS.INDEXES I
       ON I.OBJECT_ID = SS.OBJECT_ID
            AND I.INDEX_ID = SS.INDEX_ID
    INNER JOIN    SYS.OBJECTS O on O.OBJECT_ID=I.OBJECT_ID
WHERE DATABASE_ID = DB_ID()
  AND OBJECTPROPERTY (SS.OBJECT_ID,'IsUserTable') = 1
 AND  SS.OBJECT_ID = o.object_id--OBJECT_ID(@TABLENAME)
ORDER BY
       OBJECT_NAME(SS.OBJECT_ID),USER_SEEKS
    , USER_SCANS
    , USER_LOOKUPS
      , USER_UPDATES ASC

The same result can be obtained for a specific table using the below query. Replace DBNAME & TABLENAME in the code below:

USE DBNAME
GO
SELECT DB_NAME(DATABASE_ID) AS [DATABASE NAME]
     , OBJECT_NAME(SS.OBJECT_ID) AS [OBJECT NAME]
     , I.NAME AS [INDEX NAME]
     , USER_SEEKS AS [NUMBER OF SEEKS]
     , USER_SCANS AS [NUMBER OF SCANS]
     , USER_LOOKUPS AS [NUMBER OF BOOKMARK LOOKUPS]
     , USER_UPDATES AS [NUMBER OF UPDATES]
FROM SYS.DM_DB_INDEX_USAGE_STATS SS
     INNER JOIN SYS.INDEXES I ON I.OBJECT_ID = SS.OBJECT_ID AND I.INDEX_ID = SS.INDEX_ID
     INNER JOIN SYS.OBJECTS O ON O.OBJECT_ID = I.OBJECT_ID
WHERE DATABASE_ID = DB_ID()
  AND OBJECTPROPERTY(SS.OBJECT_ID, 'IsUserTable') = 1
  AND O.name = 'TABLENAME'  -- Add this condition to filter for the specific table
ORDER BY OBJECT_NAME(SS.OBJECT_ID), USER_SEEKS, USER_SCANS, USER_LOOKUPS, USER_UPDATES ASC;

Tuesday, September 1, 2009

Connect to SSIS remotely and edit default location where packages are stored


There may be situations when some users have to connect to SSIS remotely form the workstations because users cannot be logging into the actual servers. There are some steps that need to be done for having users to successfully connect to SSIS. Else they may get an error message

Perform the below steps to change the configuration of how SSIS accepts remote connections in a cluster


Note that the first step is only required for those instances in a cluster and it did not require me to make these changes to connect to a server not in cluster.
  1. Search for the MsDTSsrvr.ini file on the server and open it with notepad:
The content of the .ini file:
<?xml version="1.0" encoding="utf-8"?>
<DtsServiceConfiguration xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
  <StopExecutingPackagesOnShutdown>true</StopExecutingPackagesOnShutdown>
  <TopLevelFolders>
    <Folder xsi:type="SqlServerFolder">
      <Name>MSDB</Name>
      <ServerName>servername\instancename</ServerName>
    </Folder>
    <Folder xsi:type="FileSystemFolder">
      <Name>File System</Name>
      <StorePath>M:\MSSQL$SQL3\SSISPackages</StorePath>
    </Folder>
  </TopLevelFolders> 
</DtsServiceConfiguration>
Edit the highlighted ones in red as required.
The store path is by default D:\AppWin32\MSSQL\90\DTS\Packages or C:\Program Files\Microsoft SQL Server\90\DTS\Packages unless you wish to modify like in the above case. That location is where the packages get stored when you import into a file system and not msdb.
When we import any package into the msdb d/b there is no physical .dtsx file stored on the server.
  1. Add the user/group to Distributed Com group on the server
  1. Go to Admin tools -- Component services -- Computers -- My computer -- DComConfig -- MsDtsSrvr, right click on it and hit properties
  1. Go to ‘Security’ tab and edit the first 2 sections: ‘Launch and Activation Permissions’ and ‘Access Permissions’ and hit edit to add/remove users as required.

Execute an SSIS package under a proxy account using SQL Agent job

Database administrators might sometimes come across the situations where the users need access to run SSIS packages (in environments where the DBA team does not generally own the responsibility of ancillary services). We can’t let the users have admin privileges to be able to run the packages as jobs. In such situations proxy account comes into play. A proxy account can be created and the job can be configured to be run under that proxy account. Follow the below steps to accomplish that

1. A new NT account needs to be created for the purpose of having a SQL job (SSIS package) executed under a proxy account. Most important the NT account has to be created with ‘No Password Expiry’ option.

2. Provide the SQL DBA team a list of users that need to have access and execute the jobs on SQL Server.

3. Add the userid (that needs to execute the jobs) provided by the Business Unit to SQLAgentUserRole in the msdb database. Also make that userid owner for the job that needs to be executed.

Creating a new proxy by linking it to an already existing credential:

4. Grant access to the above created NT account in this case on SQL Server with sysadmin permissions. (creating a new login).

5. Create a new credential:

Expand security --> go to credentials --> right-click and hit new credential.

Put in the password for the already existing NT account which has sysadm permissions on the SQL server

6. Expand SQL Server Agent --> right click on Proxies and hit new proxy

On ‘General’ tab enter the proxy name and enter the credential name created in the first step select the subsystems as required

7. On the ‘Principal’ hit on the button ‘Add’ and add the required logins that are required to make use of the proxy account.

8. Edit the job properties and hit ‘Edit’.

In the drop down list of ‘Run as’ select the newly created proxy account and hit ok.

The job should run successfully now.

Wednesday, August 26, 2009

Execute sp_msforeachdb by excluding the system databases

For executing a query against each database, we can use either dynamic SQL (search my other posts for info on that) or use undocumented stored procedure sp_msforeachdb. It is particularly helpful when we want to do this against multiple databases.

The other challenge in using this is excluding the system databases if we want the query to be run just against the user databases. Use the below sql statements to accomplish that. Consider executing a sql statement of sp_addrole 'rolename' against each database on just the user databases.

EXEC master..sp_MSForeachdb
'USE [?]
IF DB_ID(''?'') > 4
exec sp_addrole ''rolename'''

Similar commands like above can be used to force an update statistics on all the tables in a given database:

EXEC sp_MSForEachTable  "update statistics ? with fullscan"

Tuesday, August 25, 2009

Threshold of Performance counter values to be considered as ideal

Performance monitor has a set of counters that can be monitored over a given period of time to see if the Windows server that is hosting the Sql Server is in good health.

A general question for a SQL Server DBA while monitoring counters in performance monitor is 'What is the threshold for each of the counters that can be considered as alarm bells for further investigation?'

The below table summarizes the counters and their ideal values. It was given to me by a Microsoft Engineer who works at our client place. All the values have been put together by his team at Microsoft.

The below should be a very good starting point for a DBA to either zoom in or rule out OS as the culprit.

PerformanceObject

Counter

Observed value

Threshold

Memory

Available MBytes

<10mb>

Memory

Cache Bytes

Memory

Commit Limit

Memory

Committed Bytes

> Physical Bytes is bad

Memory

Free System Page Table Entries

<5,000>very bad; <7,000>

Memory

Page Faults/sec

<>

Memory

Pages/sec

>2500/s bad for modern servers; less for older servers

Memory

Pool NonPaged Bytes

32-bit: 100MB-260MB; 64-bit: 1GB-128GB (depends on total physical RAM; is it close to being consumed)

Memory

Pool Paged Bytes

32-bit: 120MB-480MB; 64-bit: 1GB-128GB (depends on total physical RAM; is it close to being consumed)

Memory

System Cache Resident Bytes

Network Interface

Bytes total/sec

*

look for descending numbers to indicate network problems (kk)

Paging File

% Usage

*

>70% is a cause for concern

Paging File

% Usage Peak

*

PhysicalDisk/LogicalDisk

% Disk Time

*

>85% (consider spindle count)

PhysicalDisk/LogicalDisk

% Free Space

*

10-30%

PhysicalDisk/LogicalDisk

% Idle Time

*

PhysicalDisk/LogicalDisk

Avg. Disk Bytes/Read

*

PhysicalDisk/LogicalDisk

Avg. Disk Bytes/Transfer

*

Avg. I/O size; theoretical ~90-100MB/s sequential; reality significantly less

PhysicalDisk/LogicalDisk

Avg. Disk Bytes/Write

*

PhysicalDisk/LogicalDisk

Avg. Disk Queue Length

*

1.5 x # of spindles

PhysicalDisk/LogicalDisk

Avg. Disk Read Queue Length

*

1.5 x # of spindles

PhysicalDisk/LogicalDisk

Avg. Disk sec/Read

*

Disk Latency > 50ms POOR

PhysicalDisk/LogicalDisk

Avg. Disk sec/Transfer

*

Disk Latency > 50ms POOR

PhysicalDisk/LogicalDisk

Avg. Disk sec/Write

*

Disk Latency > 50ms POOR

PhysicalDisk/LogicalDisk

Avg. Disk Queue/sec

*

Disk Latency

PhysicalDisk/LogicalDisk

Avg. Disk Write Queue Length

*

1.5 x # of spindles

PhysicalDisk/LogicalDisk

Current Disk Queue Length

*

1.5 x # of spindles

PhysicalDisk/LogicalDisk

Disk Bytes/sec

*

PhysicalDisk/LogicalDisk

Disk Read Bytes/sec

*

PhysicalDisk/LogicalDisk

Disk Write Bytes/sec

*

PhysicalDisk/LogicalDisk

Disk Reads/sec

*

}~150 (for random)

PhysicalDisk/LogicalDisk

Disk Transfers/sec

*

}

PhysicalDisk/LogicalDisk

Disk Writes/sec

*

}

PhysicalDisk/LogicalDisk

Split IO/sec

*

Fewer the better

Process

% Processor Time

_Total,

Process

Page Faults/sec

_Total,

<>

Process

Handle Count

_Total,

Process

IO Data Operations/sec

_Total,

Process

IO Other Operations/sec

_Total,

Process

Private Bytes

_Total,

Process

Virtual Bytes

_Total,

Process

Working Set

_Total,

Processor

% Idle

Processor

% Privileged Time

Processor

% Processor Time

Processor

% User

User+Priv>80% CPU bottleneck

Processor

Interrupts/sec

System

Context Switches/sec

5,000/s x # of CPUs; somewhat flexible if other CPU-related counters are good--including low SQL compiles/s

System

Processor Queue Length

2 x # of CPUs; threads waiting to be executed; >2 means CPU bottleneck

Access Methods

Full Scans/sec

(Index Searches/sec)/(Full Scans/sec) > 1000

Access Methods

Index Searches/sec

(Index Searches/sec)/(Full Scans/sec) > 1000

Access Methods

Mixed Page Allocations/sec

baseline

Access Methods

Page Splits/sec

baseline or <>

Access Methods

Mixed Page Allocations/sec

baseline

Access Methods

Workfiles Created/sec

baseline

Access Methods

Worktables Created/sec

baseline

Access Methods

Worktables from Cache Ratio

baseline

General Statistics

Temp Tables Creation Rate

baseline

General Statistics

Temp Tables for Destruction

baseline

Buffer Manager

Buffer Cache Hit Ratio

98% OK, 99% better, 99.8% desirable

Buffer Manager

Checkpoint pages/sec

baseline

Buffer Manager

Database Pages

<>

Buffer Manager

Lazy writes/sec

baseline

Buffer Manager

Page Life Expectancy

<300s>1,000 or >10,000s much better

Buffer Manager

Procedure Cache Pages

(2000) prefer stable values, otherwise cache thrash may be occurring

Plan Cache

Cache Pages

(2005) prefer stable values, otherwise cache thrash may be occurring

Buffer Manager

Total pages

baseline

Cache Manager

Cache Hit Ratio

<99.9>98 not optimal

Databases

Data File(s) Size (kb)

Databases

Log Flushes/sec

baseline

Databases

Percent Log Used

Databases

Transactions/sec

baseline

General Statistics

Logins/sec & Logouts/sec

baseline; >2/sec suggests app's implementation of connection pooling may be flawed

General Statistics

User Connections

~

Latches

Average Latch Wait Time (ms)

~

Latches

Latch Waits/sec

(Total Latch Wait Time) / (Latch Waits/Sec) <>

Latches

Total Latch Wait Time (ms)

(Total Latch Wait Time) / (Latch Waits/Sec) <>

Locks

Avg. Wait Time (ms)

~

Locks

Lock Timeouts/sec

> 0

Locks

Lock Wait Time (ms)

Locks

Lock Waits/sec

> 0

Locks

Number of Deadlocks/sec

> 0

Memory Manager

Granted Workspace Memory (KB)

baseline

Memory Manager

Memory Grants Pending

>0 is cause for concern

Memory Manager

Memory Grants Outstanding

Memory Manager

Target Server Memory

high or rising value indicates insufficient memory

Memory Manager

Total Server Memory

~=Physical Ram

SQL Statistics

Batch Requests/sec

OLTP workloads: server should support ~2,500/s/core

SQL Statistics

SQL Compilations/sec

if compiles > 10% of Batch Requests, then app may not be caching effectively

SQL Statistics

SQL Recompilations/sec

Recompiles > 10% of Compilations should be investigated

SQL Server:Buffer Manager

Free list stalls/sec

<>

SQL Server:Buffer Manager

Free pages

> 640

SQL Server:Buffer Manager

Lazy Writes/Sec

<>

SQLServer:Buffer Manager

Page lookups/sec

(Page lookups/sec) / (Batch Requests/sec) <>

SQL Server:Buffer Manager

Page reads/sec

<>

SQL Server:Buffer Manager

Page writes/sec

<>


Display the list of indexes and columns for a given table

You might find yourself in a situation where you have to analyze the indexes for a given table and want to know which columns are part of wh...