web-dev-qa-db-ja.com

1つのクエリですべてのファイル、それらのフィルレベル、および追加のドライブ情報をクエリするにはどうすればよいですか?

次の結果をまとめるクエリが必要です。

  • すべてのデータベースのデータベースファイルごとに1行
  • ファイルが置かれている場所のドライブに関する情報を含む追加の列
  • ファイルのサイズと使用法に関する情報

これまで、次のクエリをまとめました。

    SELECT 
     GETDATE() as dt
     ,@@SERVERNAME as srv
     ,F.name 
     ,F.physical_name    
     ,Round(F.size * 8 / 1024, 2) as FileSizeMb
    , CAST(FILEPROPERTY(F.name, 'SpaceUsed') AS INT)/128 as FileUsedMB
    ,(F.size/128 - CAST(FILEPROPERTY(F.name, 'SpaceUsed') AS INT)/128) AS FileFreeMB
    ,Convert(decimal(18,2), (F.size/128 - CAST(FILEPROPERTY(F.name, 'SpaceUsed') AS INT)/128) / (F.size * 8 / 1024.1) * 100) as SpaceFreePerc 
     ,stat.size_on_disk_bytes / 1024 / 1024 SizeOnDiskMb
     ,drv.volume_mount_point, drv.logical_volume_name, drv.available_bytes, drv.total_bytes 
 FROM sys.master_files F 
 inner join sys.dm_io_virtual_file_stats(NULL, NULL) stat on F.database_id = stat.database_id AND F.file_id = stat.file_id 
 CROSS APPLY sys.dm_os_volume_stats(F.database_id, F.FILE_ID) drv

今、私はここで2つの問題に直面しています。

  1. FILEPROPERTY SpaceUsedは、現在のコンテキストのデータベースでのみ使用できます。したがって、すべてのデータベース/ファイルからすべてのFilepropertiesを収集するソリューションが必要です。
  2. CROSS APPLY dm_os_volume_statsは、互換性レベル80のデータベースでは機能しませんが、いくつかは配置されています。

与えられた条件でこの目標を達成する方法はありますか?とにかく最終的にはストアドプロシージャの一部になるため、ソリューションは1ステートメントのクエリである必要はありません。

4
Magier

CROSS APPLYを回避するために私が考えることができる最も簡単な方法は、sys.dm_os_volume_statsdatabase_idの明示的なパラメーターを使用してfile_idを呼び出すことです。これは、すべてのdb /ファイルコンボに対して単一行の結果を実行することを意味します。

まず、結果を保持する#tempテーブルを作成します。

CREATE TABLE #x(dt datetime, srv nvarchar(520), logical_name sysname,
  physical_name sysname, FileSizeMb int, FileUsedMB int, FileFreeMB int,
  SpaceFreePerc decimal(18,2), SizeOnDiskMB int, volume_mount_point nvarchar(256), 
  logical_volume_name nvarchar(256), available_bytes bigint, total_bytes bigint);

さて、いくつかの変数とカーソル。ここでの唯一のトリッキーな部分は、実際には@exec変数です。これにより、動的SQLの各反復を適切なデータベースコンテキストで実行できます。

DECLARE @database_id int, @file_id int, @logical_name sysname, 
  @physical_name nvarchar(520), @size int, @sql nvarchar(max), @exec sysname;

DECLARE c CURSOR LOCAL FAST_FORWARD
  FOR SELECT database_id, [file_id], name, physical_name, size
  FROM sys.master_files;

SET @sql = N'SELECT 
     GETDATE() as dt ,@@SERVERNAME as srv, @logical_name, @physical_name    
     ,Round(@size * 8 / 1024, 2) as FileSizeMb
     , CAST(FILEPROPERTY(@logical_name, N''SpaceUsed'') AS INT)/128 as FileUsedMB
     ,(@size/128 - CAST(FILEPROPERTY(@logical_name, N''SpaceUsed'') AS INT)/128) AS FileFreeMB
     ,Convert(decimal(18,2), (@size/128 - CAST(FILEPROPERTY(@logical_name, N''SpaceUsed'') AS INT)/128) / (@size * 8 / 1024.1) * 100) as SpaceFreePerc
     ,stat.size_on_disk_bytes / 1024 / 1024 SizeOnDiskMb
     ,drv.volume_mount_point, drv.logical_volume_name, drv.available_bytes, drv.total_bytes 
 FROM sys.master_files F 
 inner join sys.dm_io_virtual_file_stats(NULL, NULL) stat on F.database_id = stat.database_id AND F.file_id = stat.file_id 
 CROSS JOIN sys.dm_os_volume_stats(@database_id, @file_id) drv
 WHERE F.database_id = @database_id AND F.file_id = @file_id;';

OPEN c;
FETCH NEXT FROM c INTO @database_id, @file_id, @logical_name, @physical_name, @size;

WHILE (@@FETCH_STATUS <> -1)
BEGIN
  SET @exec = DB_NAME(@database_id) + N'.sys.sp_executesql ';    
  INSERT #x EXEC @exec @sql, N'@database_id int, @file_id int, 
    @logical_name sysname, @physical_name nvarchar(520), @size int',
    @database_id, @file_id, @logical_name, @physical_name, @size;

  FETCH NEXT FROM c INTO @database_id, @file_id, @logical_name, @physical_name, @size;
END

SELECT * FROM #x;

CLOSE c; DEALLOCATE c;
6
Aaron Bertrand