-
Notifications
You must be signed in to change notification settings - Fork 44
/
Copy pathautomated index rebuild.sql
573 lines (470 loc) · 22.7 KB
/
automated index rebuild.sql
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
/*
Automated index maint script
Last Update: 20200801
Primary objective is to never perform an offline index rebuild unless it is the only option, and desired. This is commented out by default.
Loops past errors, throws an error at the end if any errors were recorded.
IMPORTANT: When scheduling in a SQL Agent job, with each database as a job step, set each job step to "go to the next step" on failure.
#TODO
1. Check the name of the DBALogging/DBAAdmin/DBAHound database local to this SQL instance and replace if necessary.
2. Change @StartWindow and @EndWindow values as desired.
3. Set the job step database context correctly - this script doesn't have a USE. For example, in the database property of the SQL Agent job step.
4. If upgrading this script, look at the bottom to see if the Logging table needs data type changes. We changed the timestamp field data types to datetimeoffset a couple years ago and may need to update the logging table.
Note recent data type changes to the IndexMaintLog table, if it already exists. See comment block at bottom, including a conversion from old datetime to datetimeoffset data.
5. IMPORTANT! Change @TestMode to 0 when ready for production
#Optional TODO's
1. Change name in ALTER ... BULK_LOGGED below if desired to use this. Change name in ALTER ... FULL near end if desired to use this. Commented out by default because this isn't feasible in many situations.
2. Enable @CompressWhenPossible to start enabling data_compression = PAGE everywhere. Not recommended without testing.
*/
DECLARE @TestMode bit, @StartWindow tinyint, @EndWindow tinyint, @CompressWhenPossible bit;
SELECT @TestMode = 0 -- TODO: flip to 1 to run out of cycle, without actually executing any code.
SELECT @StartWindow = 1 -- TODO: 1AM Range (0-23). 24-hour of the day. Ex: 4 = 4am, 16 = 4pm. 0 = midnight.
SELECT @EndWindow = 24 -- TODO: 5AM Range (0-23). 24-hour of the day. Ex: 4 = 4am, 16 = 4pm. 0 = midnight.
SELECT @CompressWhenPossible = 0 -- 0 = Don't introduce new compression on indexes. 1= Compress when possible, even currently uncompressed databases.
IF @TestMode = 1 print '--Test mode is ON'
DECLARE @maxlogid int --used later on to throw an error if the loop encountered errors at any point
SELECT @maxlogid = ISNULL(MAX(id),0) From DBALogging.dbo.IndexMaintLog
SET XACT_ABORT ON;
BEGIN TRY
IF EXISTS (select state_desc from sys.databases d where database_id = db_id() and (state_desc <> 'online' or is_read_only = 1))
THROW 51000, '--This database is not online, skipping', 1; --stop if this database isn't online or is read_only. It may have already failed before this point anyway.
IF EXISTS (SELECT dc.database_name
FROM sys.dm_hadr_availability_replica_states rs
inner join sys.availability_databases_cluster dc on rs.group_id = dc.group_id
WHERE is_local = 1 and role_desc <> 'PRIMARY' and dc.database_name = db_name()
)
THROW 51000, '--This database is a secondary replica currently, skipping', 1; --stop if this database is a secondary replica in an AG
--IF @TestMode = 0
--ALTER DATABASE wideWorldImporters SET RECOVERY BULK_LOGGED; --#TODO Change database name
SET NOCOUNT ON;
DECLARE @objectid int
, @indexid int
, @partitioncount bigint
, @SchemaName sysname
, @ObjectName sysname
, @indexname sysname
, @partitionnum bigint
, @partitions bigint
, @frag float
, @Command varchar(8000)
, @Can_Reorg bit
, @Can_RebuildOnline bit
, @Can_Compress bit
, @ProductVersion varchar(15)
, @ServerEdition varchar(30)
, @type_desc sysname
SET @ServerEdition = convert(varchar(30),(SELECT SERVERPROPERTY('Edition')))
SET @ProductVersion = convert(varchar(15),(SELECT SERVERPROPERTY('ProductVersion')))
-- ensure the temporary table does not exist
IF EXISTS (SELECT name FROM tempdb.sys.objects WHERE name like '%C__work_to_do%')
DROP TABLE #C__work_to_do;
CREATE TABLE #C__work_to_do
(
objectid int NOT NULL
, indexid int NOT NULL
, partitionnum int NOT NULL
, frag decimal(9,2) NOT NULL
, Can_Reorg bit NOT NULL
, Can_RebuildOnline bit NOT NULL
, Can_Compress bit NOT NULL
, indexname sysname NOT NULL
, [type_desc] sysname NOT NULL
, partitioncount int NOT NULL,
PRIMARY KEY ( objectid, indexid, partitionnum )
)
-- conditionally select from the function, converting object and index IDs to names.
INSERT INTO #C__work_to_do
( objectid
, indexid
, partitionnum
, frag
, Can_Reorg
, Can_RebuildOnline
, Can_Compress
, indexname
, [type_desc]
, partitioncount
)
SELECT DISTINCT
objectid = s.object_id
, indexid = s.index_id
, partitionnum = s.partition_number
, frag = max(s.avg_fragmentation_in_percent)
, Can_Reorg = CASE WHEN i.type_desc like '%columnstore%' THEN 1 ELSE i.allow_page_locks END --An index cannot be reorganized when allow_page_locks is set to 0.
, Can_RebuildOnline =
CASE
WHEN pa.partitioncount > 1 and left(@ProductVersion,2) <= 11 --only starting with SQL2014 online partitioned index operations are supported
THEN 0
WHEN A.user_type_id is not null
THEN 0 -- Cannot do ONLINE REBUILDs with certain data types in the index (key or INCLUDE).
WHEN i.type_desc in ('xml','spatial') THEN 0 -- Cannot do ONLINE REBUILDs for certain index types.
WHEN i.type_desc like '%columnstore%' THEN 0 -- cannot rebuild columnstores
WHEN (left(@ProductVersion,2) >= 10 ) and (@ServerEdition like 'Developer%' or @ServerEdition like 'Enterprise%' )
THEN 1
ELSE 0
END
, Can_Compress = CASE
WHEN @CompressWhenPossible = 0 THEN 0 --override and don't do compression if it's turned off
WHEN (left(@ProductVersion,2) >= 10 ) and (@ServerEdition like 'Developer%' or @ServerEdition like 'Enterprise%' )
THEN 1
ELSE 0
END
, indexname = i.name --select object_name(object_id), *
, i.[type_desc]
, pa.partitioncount
FROM sys.indexes i
inner join sys.partitions p on p.index_id = i.index_id and p.object_id = i.object_id
cross apply
sys.dm_db_index_physical_stats (DB_ID(), i.object_id, i.index_id , p.partition_number, 'LIMITED') s
inner join
(select partitioncount = count (*), object_id, index_id from sys.partitions group by object_id, index_id) as pa
on pa.object_id = i.object_id AND pa.index_id = i.index_id
left outer join
(
select
c.object_id, index_id = ISNULL(ic.index_id, i.index_id), user_type_id = COALESCE(t.user_type_id, null)
from sys.columns c
left outer join sys.indexes i
on c.object_id = i.object_id and i.index_id <= 1 --need the heap or clustered index to be returned here, because all columns are included in the heap or clustered index
left outer join sys.index_columns ic
on ic.object_id = c.object_id
and ic.column_id = c.column_id
left outer join sys.types t
on c.user_type_id = t.user_type_id
WHERE
( t.name in (N'image', N'text', N'ntext', N'xml')
--or (t.name in ('varchar', 'nvarchar', 'varbinary') --This is no longer necessary after SQL Server 2012!
-- and c.max_length = -1 --indicates (MAX)
-- )
)
) A
on i.object_id = A.object_id
AND i.index_id = A.index_id
WHERE
1=1
AND s.avg_fragmentation_in_percent > 20.0
AND s.index_id > 0 --
AND s.page_count > 1280 --12800 pages is 100mb
and i.name is not null
group by
s.object_id
, s.index_id
, s.partition_number
, i.allow_page_locks
, A.index_id
, A.user_type_id
, i.type_desc
, i.name
, pa.partitioncount;
IF @TestMode = 1
SELECT * FROM #C__work_to_do ;
IF CURSOR_STATUS('local', 'curIndexPartitions') >= 0
BEGIN
CLOSE curIndexPartitions;
DEALLOCATE curIndexPartitions;
END
-- Declare the cursor for the list of partitions to be processed.
DECLARE curIndexPartitions CURSOR LOCAL FAST_FORWARD FOR
SELECT * FROM #C__work_to_do;
-- Open the cursor.
OPEN curIndexPartitions;
-- Loop through the partitions.
FETCH NEXT
FROM curIndexPartitions
INTO @objectid, @indexid, @partitionnum, @frag, @Can_Reorg, @Can_RebuildOnline, @Can_Compress, @indexname, @type_desc, @partitioncount;
WHILE @@FETCH_STATUS = 0
BEGIN
BEGIN TRY
DECLARE @currenthour int = datepart(hour, sysdatetimeoffset())
IF(@StartWindow > @EndWindow -- wraps midnight
AND ( @currenthour >= @StartWindow
OR (@currenthour <= @StartWindow
AND @currenthour < @EndWindow)
)
)
OR
(@StartWindow <= @EndWindow -- AM only or PM only
AND @currenthour >= @StartWindow
and @currenthour < @EndWindow
)
OR
@TestMode = 1
BEGIN --Begin Time frame
SELECT @ObjectName = o.name
, @SchemaName = s.name
FROM sys.objects AS o
INNER JOIN sys.schemas as s ON s.schema_id = o.schema_id
WHERE o.object_id = @objectid;
--SELECT @objectid, @indexid, @partitionnum, @frag, @Can_Reorg, @Can_RebuildOnline, @indexname, @ObjectName, @SchemaName
SELECT @partitioncount = count (*)
FROM sys.partitions
WHERE object_id = @objectid AND index_id = @indexid;
SELECT @Command = '';
-- 30 is an arbitrary decision point at which to switch between reorganizing and rebuilding
IF @frag > 20.0 and @Can_Reorg = 1
BEGIN
IF @TestMode = 1 print '--30%+ reorg'
SELECT @Command = 'ALTER INDEX [' + @indexname + '] ON [' + @SchemaName + '].[' + @ObjectName + '] REORGANIZE ';
IF @partitioncount > 1
SELECT @Command = @Command + ' PARTITION=' + RTRIM( CONVERT (VARCHAR(5), @partitionnum));
IF @type_desc not like '%columnstore%'
SELECT @Command = @Command + '; UPDATE STATISTICS [' + @SchemaName + '].[' + @ObjectName + '] ([' + @indexname + ']); '
END
--If greater than 50% frag, instead consider ONLINE Rebuild, falling back to a REORG (or, to an offline rebuild, commented out by default.)
--Both REBUILD ONLINE and REORG are considered "online" operations. By default, ONLINE operations only.
IF @frag >= 50.0
BEGIN
-- INDEX REBUILD with ONLINE = ON reduces the impact of locking to the production server,
-- though it will still create resource contention by keeping the drives and tempdb busy.
-- Unlike REORGANIZE steps, a REBUILD also updates the STATISTICS of an index.
IF @Can_RebuildOnline = 1 and @type_desc not like '%columnstore%'
BEGIN
IF @TestMode = 1 print '--50%+ rebuild online'
SELECT @Command = 'ALTER INDEX [' + @indexname +'] ON [' + @SchemaName + '].[' + @ObjectName + '] REBUILD ';
IF @partitioncount > 1
SELECT @Command = @Command + ' PARTITION=' + RTRIM(CONVERT (VARCHAR(5), @partitionnum))
SELECT @Command = @Command + ' WITH (ONLINE = ON'
IF @Can_Compress = 1
SELECT @Command = @Command + ', DATA_COMPRESSION = PAGE'
select @Command = @Command + ', SORT_IN_TEMPDB = ON);'
END
--REORGANIZE processes are always ONLINE and are less intense than REBUILDs.
ELSE IF @Can_Reorg = 1
BEGIN
IF @TestMode = 1 print '--50%+ reorg'
SELECT @Command = 'ALTER INDEX [' + @indexname +'] ON [' + @SchemaName + '].[' + @ObjectName + '] REORGANIZE ';
IF @partitioncount > 1
SELECT @Command = @Command + ' PARTITION=' + RTRIM( CONVERT (VARCHAR(5), @partitionnum))
IF @type_desc not like '%columnstore%'
SELECT @Command = @Command + '; UPDATE STATISTICS [' + @SchemaName + '].[' + @ObjectName + '] ([' + @indexname + ']); '
END
--OPTIONAL: Only do a full, offline index rebuild in the middle of the night.
--ELSE IF datepart(hour, sysdatetimeoffset()) < 3 --inclusive both hours.
--ELSE
--BEGIN
-- IF @TestMode = 1 print '--60%+ rebuild offline!'
-- SELECT @Command = 'ALTER INDEX [' + @indexname + '] ON [' + @SchemaName + '.' + @ObjectName + '] REBUILD';
-- IF @partitioncount > 1
-- SELECT @Command = @Command + ' PARTITION = ' + CONVERT (VARCHAR(5), @partitionnum);
-- SELECT @Command = @Command + ' WITH ('
-- IF @Can_Compress = 1
-- SELECT @Command = @Command + ' DATA_COMPRESSION = PAGE, '
-- select @Command = @Command + 'SORT_IN_TEMPDB = ON);'
--END
END
IF @Command <> ''
BEGIN
INSERT INTO DBALogging.dbo.IndexMaintLog (CurrentDatabase, Command, ObjectName, BeginTimeStamp, StartWindow, EndWindow, TestMode)
SELECT DB_NAME(), @Command, '[' + DB_Name() + '].[' + @SchemaName + '].[' + @ObjectName + ']', sysdatetimeoffset(), @StartWindow, @EndWindow, @TestMode
SELECT @Command = 'SET QUOTED_IDENTIFIER ON;
' + @Command --for a specific issue that can cause some rebuild operations to fail if this is OFF. Force it ON, even though it is ON by default.
BEGIN TRY
IF @TestMode = 0 EXEC (@Command);
IF @TestMode = 1 print N'--Executed:
' + @Command + ' --Frag level: ' + cast(@frag as varchar(10));
UPDATE DBALogging.dbo.IndexMaintLog
SET EndTimeStamp = sysdatetimeoffset()
, Duration_s = datediff(s, BeginTimeStamp, sysdatetimeoffset())
where id = SCOPE_IDENTITY() and EndTimeStamp is null
END TRY
BEGIN CATCH
IF @TestMode = 1 print N'--Error: ' + ERROR_MESSAGE()
UPDATE DBALogging.dbo.IndexMaintLog
SET ErrorMessage = cast(ERROR_NUMBER() as varchar(9)) + ' ' + ERROR_MESSAGE()
where id = SCOPE_IDENTITY() and EndTimeStamp is null
END CATCH
END
END --End Time frame
END TRY
BEGIN CATCH
IF @TestMode = 1 print N'--Failed to execute the command: ' + @Command + N' Error Message: ' + ERROR_MESSAGE()
END CATCH
FETCH NEXT FROM curIndexPartitions
INTO @objectid, @indexid, @partitionnum, @frag, @Can_Reorg, @Can_RebuildOnline, @Can_Compress, @indexname, @type_desc, @partitioncount;
END
-- Close and deallocate the cursor.
IF CURSOR_STATUS('local', N'curIndexPartitions') >= 0
BEGIN
CLOSE curIndexPartitions;
DEALLOCATE curIndexPartitions;
END
-- drop the temporary table
IF EXISTS (SELECT name FROM tempdb.sys.objects WHERE name like '%C__work_to_do%')
DROP TABLE #C__work_to_do;
--Begin smart update stats
select @currenthour = datepart(hour, sysdatetimeoffset())
IF(@StartWindow > @EndWindow -- wraps midnight
AND ( @currenthour >= @StartWindow
OR (@currenthour <= @StartWindow
AND @currenthour < @EndWindow)
)
)
OR
(@StartWindow <= @EndWindow -- AM only or PM only
AND @currenthour >= @StartWindow
and @currenthour < @EndWindow
)
OR
@TestMode = 1
BEGIN
declare @tsqllist table (id int not null identity(1,1) primary key, SchemaName sysname, ObjectName sysname, tsqltext nvarchar(4000) not null, modification_counter int not null, [rows] int not null)
insert into @tsqllist ( SchemaName, ObjectName, tsqltext, modification_counter, [rows])
SELECT distinct
s.name AS SchemaName
, o.name AS ObjectName
, tsqltext = CASE WHEN i.type_desc like '%columnstore%' THEN NULL ELSE
N'UPDATE STATISTICS '
+ QUOTENAME(s.name) + N'.' + QUOTENAME(o.name) + N' '
+ QUOTENAME(STA.name) + N' '
+ 'WITH RESAMPLE'
+ CASE WHEN
STA.is_incremental = 1 and --Only works in SQL 2014+, comment out this line in prior versions.
MAX(p.partition_number) OVER (PARTITION by STA.name, i.name) > 1 THEN ' ON PARTITIONS (' + cast(p.partition_number as varchar(5)) + ') ' ELSE '' END
END
+ CASE WHEN @TestMode = 1 THEN '--Pct updated ' + ISNULL(convert(varchar(50), convert(decimal(19,2), sp.modification_counter/sp.[rows])),'') + ', Last Updated ' + ISNULL(convert(varchar(30), sp.last_updated),'') ELSE '' END
, sp.modification_counter, sp.[rows]
FROM sys.objects o
INNER JOIN sys.stats STA ON STA.object_id = o.object_id
CROSS APPLY sys.dm_db_stats_properties (STA.object_id, STA.stats_id) sp -- Only works in SQL2008R2SP2+ or SQL2012SP1+
INNER JOIN sys.schemas AS s
ON o.schema_id = s.schema_id
LEFT OUTER JOIN sys.indexes as i
on i.index_id = STA.stats_id
and (i.type_desc not like '%columnstore%')
LEFT OUTER join sys.dm_db_partition_stats p
on (
STA.is_incremental = 1 and --Only works in SQL 2014+, comment out this line in prior versions.
p.object_id = o.object_id and
i.index_id = p.index_id
)
LEFT JOIN
(SELECT IUS.object_id
,MIN(ISNULL(IUS.last_user_update, IUS.last_system_update)) AS LastUpdate
FROM sys.dm_db_index_usage_stats AS IUS
WHERE database_id = DB_ID()
AND NOT ISNULL(IUS.last_user_update, IUS.last_system_update) IS NULL
GROUP BY IUS.object_id
) AS IUS
ON IUS.object_id = STA.object_id
WHERE o.type IN ('U', 'V') -- only user tables and views
AND
(
((sp.modification_counter*1./sp.[rows]) > 1.0 --as many updates as rows in this table!
AND sp.[rows] > 1000 --1000 is arbitrary
)
OR
(sp.last_updated is null and sp.[rows] > 10000) --big enough tables should have stats updated, 10000 is arbitrary
OR
(DATEDIFF(d, ISNULL(sp.last_updated, N'1900-01-01') , IUS.LastUpdate) > 30 --indexes that haven''t been updated in the last month
and sp.[rows] > 10000
AND
((sp.modification_counter*1./sp.[rows]) > .3 --changes equal 30% of rows, an arbitrary line to cross
or modification_counter > 50000
)
)
)
OPTION (MAXDOP 1);
IF @TestMode = 1 select * from @tsqllist;
declare @s int = 1, @scount int = null, @runtsql nvarchar(4000) = null
select @scount = max(id) from @tsqllist l
IF @TestMode = 1 print '--beginning stats'
while (@s <= @scount)
BEGIN
--Check that we are still in time frame
select @currenthour = datepart(hour, sysdatetimeoffset())
IF(@StartWindow > @EndWindow -- wraps midnight
AND ( @currenthour >= @StartWindow
OR (@currenthour <= @StartWindow
AND @currenthour < @EndWindow)
)
)
OR
(@StartWindow <= @EndWindow -- AM only or PM only
AND @currenthour >= @StartWindow
and @currenthour < @EndWindow
)
OR
@TestMode = 1
BEGIN
--actually executes the scripts.
select @runtsql = tsqltext
, @ObjectName = ObjectName
, @SchemaName = SchemaName
from @tsqllist where id = @s
INSERT INTO DBALogging.dbo.IndexMaintLog (CurrentDatabase, Command, ObjectName, BeginTimeStamp, StartWindow, EndWindow, TestMode)
SELECT DB_NAME(), @runtsql, '[' + DB_Name() + '].[' + @SchemaName + '].[' + @ObjectName + ']', sysdatetimeoffset(), @StartWindow, @EndWindow, @TestMode
BEGIN TRY
IF @TestMode = 0 EXEC (@runtsql);
IF @TestMode = 1 PRINT N'--Executed:
' + @runtsql
UPDATE DBALogging.dbo.IndexMaintLog
SET EndTimeStamp = sysdatetimeoffset()
, Duration_s = datediff(s, BeginTimeStamp, sysdatetimeoffset())
where id = SCOPE_IDENTITY() and EndTimeStamp is null
END TRY
BEGIN CATCH
IF @TestMode = 1 print N'--Error: ' + ERROR_MESSAGE()
UPDATE DBALogging.dbo.IndexMaintLog
SET ErrorMessage = cast(ERROR_NUMBER() as varchar(9)) + ' ' + ERROR_MESSAGE()
WHERE id = SCOPE_IDENTITY() and EndTimeStamp is null
END CATCH
END
set @s = @s + 1
END
IF @TestMode = 1 print '--end update stats'
END
--IF @TestMode = 0
--ALTER DATABASE [WideWorldImporters] SET RECOVERY FULL --#TODO Change database name
END TRY
BEGIN CATCH
IF @TestMode = 1 print N'--Failed to execute. Error Message: ' + ERROR_MESSAGE()
IF ERROR_NUMBER() <> 51000 --detect a user-defined short-circuit above, for an offline database or a secondary replica, for example. This should not cause the job to fail.
BEGIN
--Write a record to the logging table, while we'll use below to THROW an error and cause the job to fail if >=1 errors are found.
INSERT INTO DBALogging.dbo.IndexMaintLog (CurrentDatabase, ErrorMessage , BeginTimeStamp, TestMode, ObjectName, Command)
SELECT DB_NAME(), cast(ERROR_NUMBER() as varchar(9)) + ' ' + ERROR_MESSAGE(), sysdatetimeoffset(), @TestMode, '[' + DB_Name() + '].[' + @SchemaName + '].[' + @ObjectName + ']', coalesce(@Command, @runtsql)
END
IF EXISTS (SELECT name FROM tempdb.sys.objects WHERE name like '%C__work_to_do%')
DROP TABLE #C__work_to_do;
IF CURSOR_STATUS('local', N'curIndexPartitions') >= 0
BEGIN
CLOSE curIndexPartitions;
DEALLOCATE curIndexPartitions;
END
END CATCH
IF @TestMode = 1 print '--Done'
IF @TestMode = 0 AND EXISTS (SELECT 1 from DBALogging.dbo.IndexMaintLog where id > @maxlogid and ErrorMessage is not null)
BEGIN
DECLARE @errorcount int, @errormessage varchar(1000);
SELECT @errorcount = count(ErrorMessage) from DBALogging.dbo.IndexMaintLog where id > @maxlogid and ErrorMessage is not null
SELECT @errormessage = cast(@errorcount as varchar(100)) + ' error(s) detected while performing index maintainence. Review the findings in table: SELECT * FROM DBALogging.dbo.IndexMaintLog where id > '+ cast(@maxlogid as varchar(100));
THROW 51000, @errormessage,0; --This is the only error that will rise to the SQL Agent Job.
END
IF @TestMode = 1 print '--Done'
GO
/*
DROP TABLE DBALogging.dbo.IndexMaintLog;
CREATE TABLE DBALogging.dbo.IndexMaintLog
( id int not null identity(1,1) PRIMARY KEY
, CurrentDatabase sysname not null DEFAULT (DB_NAME())
, Command nvarchar(1000) null
, ObjectName nvarchar(255) null
, BeginTimeStamp datetimeoffset(2) null
, TestMode bit null
, StartWindow tinyint null
, EndWindow tinyint null
, EndTimeStamp datetimeoffset(2) null
, Duration_s int null
, ErrorMessage nvarchar(4000) null
)
--For previous installs, we need to update the timestamp field data types in the table.
alter table DBALogging.dbo.IndexMaintLog
alter column BeginTimeStamp datetimeoffset(2) null
alter table DBALogging.dbo.IndexMaintLog
alter column EndTimeStamp datetimeoffset(2) null
--To convert data to proper time zone:
update DBALogging.dbo.IndexMaintLog
set
BeginTimeStamp = dateadd(hh, +4, BeginTimeStamp AT TIME ZONE 'Eastern Standard Time') --TODO: Check correct time zone!
, EndTimeStamp = dateadd(hh, +4, BeginTimeStamp AT TIME ZONE 'Eastern Standard Time') --TODO: Check correct time zone!
--, duration_s = duration_s - (4*60*60)
*/