జెంకిన్స్‌లో SQL సర్వర్ యొక్క ఆటోమేషన్: ఫలితాన్ని అందంగా అందిస్తుంది

మళ్ళీ అమరిక యొక్క థీమ్‌ను కొనసాగించడం జీరో టచ్ PROD RDS కింద. భవిష్యత్ DBAలు PROD సర్వర్‌లకు నేరుగా కనెక్ట్ కాలేవు, కానీ వాటిని ఉపయోగించగలవు జెంకిన్స్ పరిమిత కార్యకలాపాల కోసం ఉద్యోగాలు. DBA ఉద్యోగాన్ని ప్రారంభించింది మరియు కొంత సమయం తర్వాత ఈ ఆపరేషన్ పూర్తయినట్లు నివేదికతో కూడిన లేఖను అందుకుంటుంది. ఈ ఫలితాలను వినియోగదారుకు అందించడానికి మార్గాలను చూద్దాం.

జెంకిన్స్‌లో SQL సర్వర్ యొక్క ఆటోమేషన్: ఫలితాన్ని అందంగా అందిస్తుంది

సాధారణ అక్షరాల

చాలా చిన్నవిషయంతో ప్రారంభిద్దాం. మొదటి పద్ధతి చాలా సులభం, దాని గురించి మాట్లాడటానికి నిజంగా ఏమీ లేదు (రచయిత ఇకపై ఫ్రీస్టైల్ జాబ్‌లను ఉపయోగిస్తాడు):

జెంకిన్స్‌లో SQL సర్వర్ యొక్క ఆటోమేషన్: ఫలితాన్ని అందంగా అందిస్తుంది

sqlcmd ఏదైనా చేస్తుంది మరియు మేము దానిని వినియోగదారుకు అందజేస్తాము. ఉదాహరణకు, బ్యాకప్ ఉద్యోగాలకు అనువైనది:

జెంకిన్స్‌లో SQL సర్వర్ యొక్క ఆటోమేషన్: ఫలితాన్ని అందంగా అందిస్తుంది

RDS బ్యాకప్/పునరుద్ధరణ కింద అసమకాలికమని మర్చిపోవద్దు, కాబట్టి మీరు దాని కోసం వేచి ఉండాలి:

declare @rds table
  (id int, task_type varchar(128), database_name sysname, pct int, duration int, 
   lifecycle varchar(128), taskinfo varchar(max) null, 
   upd datetime, cre datetime,
   s3 varchar(256), ovr int, KMS varchar(256) null)
waitfor delay '00:00:20' 
insert into @rds exec msdb.dbo.rds_task_status @db_name='{db}'
select @xid=max(id) from @rds

again:
waitfor delay '00:00:02'
delete from @rds
insert into @rds exec msdb.dbo.rds_task_status @db_name='{db}' 
# {db} substituted with db name by powershell
select @stat=lifecycle,@info=taskinfo from @rds where id=@xid
if @stat not in ('ERROR','SUCCESS','CANCELLED') goto again

రెండవ పద్ధతి, CSV

ఇక్కడ ప్రతిదీ కూడా చాలా సులభం:

జెంకిన్స్‌లో SQL సర్వర్ యొక్క ఆటోమేషన్: ఫలితాన్ని అందంగా అందిస్తుంది

అయినప్పటికీ, CSVలో అందించబడిన డేటా "సరళమైనది" అయితే మాత్రమే ఈ పద్ధతి పని చేస్తుంది. మీరు ఈ విధంగా TOP N CPU ఇంటెన్సివ్ క్వెరీల జాబితాను తిరిగి ఇవ్వడానికి ప్రయత్నిస్తే, ప్రశ్న వచనంలో ఏవైనా అక్షరాలు - కామాలు, కోట్‌లు మరియు లైన్ బ్రేక్‌లు కూడా ఉండవచ్చు అనే వాస్తవం కారణంగా CSV "క్షీణిస్తుంది". అందువల్ల, మనకు మరింత సంక్లిష్టమైన విషయం అవసరం.

HTML లో అందమైన సంకేతాలు

నేను మీకు వెంటనే కోడ్ స్నిప్పెట్ ఇస్తాను

$Header = @"
<style>
TABLE {border-width: 1px; border-style: solid; border-color: black; border-collapse: collapse;}
TH {border-width: 1px; padding: 3px; border-style: solid; border-color: black; background-color: #6495ED;}
TD {border-width: 1px; padding: 3px; border-style: solid; border-color: black;}
</style>
"@
  
$Result = invoke-Sqlcmd -ConnectionString $jstr -Query "select * from DbInv" `
  | Select-Object -Property * -ExcludeProperty "ItemArray", "RowError", "RowState", "Table", "HasErrors"
if ($Result -eq $null) { $cnt = 0; }
elseif ($Result.getType().FullName -eq "System.Management.Automation.PSCustomObject") { $cnt = 1; }
else { $cnt = $Result.Rows.Count; } 
if ($cnt -gt 0) {
  $body = "<h2>My table</h2>"
  $Result | ConvertTo-HTML -Title "Rows" -Head $header -body $body `
    | Out-File "res.log" -Append -Encoding UTF8
  } else {
    "<h3>No data</h3>" | Out-File "res.log" -Append -Encoding UTF8
  }

మార్గం ద్వారా, System.Management.Automation.PSCustomObjectతో ఉన్న లైన్‌కు శ్రద్ద, గ్రిడ్‌లో సరిగ్గా ఒక లైన్ ఉంటే అది మాయాజాలం; అంతర్జాలం నుండి పరిష్కారం అంతగా అవగాహన లేకుండా తీసుకోబడింది. ఫలితంగా, మీరు అవుట్‌పుట్‌ని ఇలా ఫార్మాట్ చేస్తారు:

జెంకిన్స్‌లో SQL సర్వర్ యొక్క ఆటోమేషన్: ఫలితాన్ని అందంగా అందిస్తుంది

గ్రాఫ్‌లు గీయడం

హెచ్చరిక: క్రింద కింకీ కోడ్!
SQL సర్వర్‌లో చివరి N నిమిషాల పాటు CPUని ప్రదర్శించే ఒక ఫన్నీ ప్రశ్న ఉంది - కామ్రేడ్ మేజర్ ప్రతిదీ గుర్తుంచుకున్నట్లు తేలింది! ఈ క్విజ్‌ని ప్రయత్నించండి:

DECLARE @ts_now bigint = (SELECT cpu_ticks/(cpu_ticks/ms_ticks) 
  FROM sys.dm_os_sys_info WITH (NOLOCK)); 
SELECT TOP(256) 
  DATEADD(ms, -1 * (@ts_now - [timestamp]), GETDATE()) AS [EventTime],
  SQLProcessUtilization AS [SQLCPU], 
  100 - SystemIdle - SQLProcessUtilization AS [OtherCPU]
FROM (SELECT record.value('(./Record/@id)[1]', 'int') AS record_id, 
  record.value('(./Record/SchedulerMonitorEvent/SystemHealth/SystemIdle)[1]', 'int') 
    AS [SystemIdle], 
  record.value('(./Record/SchedulerMonitorEvent/SystemHealth/ProcessUtilization)[1]', 'int') 
    AS [SQLProcessUtilization], [timestamp] 
  FROM (SELECT [timestamp], CONVERT(xml, record) AS [record] 
  FROM sys.dm_os_ring_buffers WITH (NOLOCK)
  WHERE ring_buffer_type = N'RING_BUFFER_SCHEDULER_MONITOR' 
    AND record LIKE N'%<SystemHealth>%') AS x) AS y 
ORDER BY 1 DESC OPTION (RECOMPILE);

ఇప్పుడు, ఈ ఫార్మాటింగ్ ఉపయోగించి ($Fragment వేరియబుల్)

<table style="width: 100%"><tbody><tr style="background-color: white; height: 2pt;">
  <td style="width: SQLCPU%; background-color: green;"></td>
  <td style="width: OtherCPU%; background-color: blue;"></td>
  <td style="width: REST%; background-color: #C0C0C0;"></td></tr></tbody>
</table>

మేము లేఖ యొక్క శరీరాన్ని ఏర్పరచవచ్చు:

$Result = invoke-Sqlcmd -ConnectionString $connstr -Query $Query `
  | Select-Object -Property * -ExcludeProperty `
  "ItemArray", "RowError", "RowState", "Table", "HasErrors"
if ($Result.HasRows) {
  foreach($item in $Result) 
    { 
    $time = $itemEventTime 
    $sqlcpu = $item.SQLCPU
    $other = $itemOtherCPU
    $rest = 100 - $sqlcpu - $other
    $f = $fragment -replace "SQLCPU", $sqlcpu
    $f = $f -replace "OtherCPU", $other
    $f = $f -replace "REST", $rest
    $f | Out-File "res.log" -Append -Encoding UTF8
    }

ఏది ఇలా ఉంటుంది:

జెంకిన్స్‌లో SQL సర్వర్ యొక్క ఆటోమేషన్: ఫలితాన్ని అందంగా అందిస్తుంది

అవును, మాన్సియర్‌కి వక్రబుద్ధి గురించి చాలా తెలుసు! ఈ కోడ్ కలిగి ఉండటం ఆసక్తికరంగా ఉంది: పవర్‌షెల్ (దానిలో వ్రాయబడింది), SQL, Xquery, HTML. మేము జావాస్క్రిప్ట్‌ను HTMLకి జోడించలేకపోవడం విచారకరం (ఇది వ్రాయడం కోసం), కానీ పైథాన్ కోడ్‌ను పాలిష్ చేయడం (ఇది SQLలో ఉపయోగించవచ్చు) ప్రతి ఒక్కరి విధి!

SQL ప్రొఫైలర్ ట్రేస్ అవుట్‌పుట్

TextData ఫీల్డ్ కారణంగా ట్రేస్ CSVకి సరిపోదని స్పష్టంగా ఉంది. కానీ అక్షరంలో ట్రేస్ గ్రిడ్‌ను ప్రదర్శించడం కూడా వింతగా ఉంటుంది - పరిమాణం కారణంగా మరియు ఈ డేటా తరచుగా తదుపరి విశ్లేషణ కోసం ఉపయోగించబడుతుంది. అందువల్ల, మేము ఈ క్రింది వాటిని చేస్తాము: మేము ద్వారా కాల్ చేస్తాము invoke-SqlCmd ఒక నిర్దిష్ట స్క్రిప్ట్, దాని లోతుల్లో ఇది చేయబడుతుంది

select 
  SPID,EventClass,TextData,
  Duration,Reads,Writes,CPU,
  StartTime,EndTime,DatabaseName,HostName,
  ApplicationName,LoginName
   from ::fn_trace_gettable ( @filename , default )  

తదుపరి, ఆన్ స్నేహితుడు DBA ద్వారా యాక్సెస్ చేయగల సర్వర్‌లో, అన్ని పేర్కొన్న నిలువు వరుసలను ఆమోదించడానికి సిద్ధంగా ఉన్న ఒక ఖాళీ టెంప్లేట్, మోడల్ ప్లేట్‌తో ట్రేసెస్ డేటాబేస్ ఉంది. మేము ఈ నమూనాను ప్రత్యేక పేరుతో కొత్త పట్టికకు కాపీ చేస్తాము:

$dt = Get-Date -format "yyyyMMdd"
$tm = Get-Date -format "hhmmss"
$tableName = $srv + "_" + $dt + "_" + $tm
$copytab = "select * into " + $tableName + " from Model"
invoke-SqlCmd -ConnectionString $tstr -Query $copytab 

మరియు ఇప్పుడు మనం దానిని ఉపయోగించి మన జాడను వ్రాయవచ్చు డేటా.SqlClient.SqlBulkCopy - నేను ఇప్పటికే పైన దీనికి ఉదాహరణ ఇచ్చాను. అవును, TextDataలో స్థిరాంకాలను మాస్క్ చేయడం కూడా మంచిది:

# mask data
foreach ($Row in $Result)
{ 
  $v = $Row["TextData"]
  $v = $v -replace "'([^']{2,})'", "'str'" -replace "[0-9][0-9]+", '999'
  $Row["TextData"] = $v
}

మేము ఒకటి కంటే ఎక్కువ అక్షరాలు ఉన్న సంఖ్యలను 999తో భర్తీ చేస్తాము మరియు ఒకటి కంటే ఎక్కువ పొడవు గల స్ట్రింగ్‌లను 'str'తో భర్తీ చేస్తాము. 0 నుండి 9 వరకు ఉన్న సంఖ్యలు తరచుగా జెండాలుగా ఉపయోగించబడతాయి మరియు మేము వాటిని తాకము, అలాగే ఖాళీ మరియు ఏక-అక్షరాల తీగలను - 'Y', 'N', మొదలైనవి తరచుగా వాటిలో కనిపిస్తాయి.

మన జీవితాలకు కొంత రంగును జోడించుకుందాం (కచ్చితంగా 18+)

పట్టికలలో, మీరు తరచుగా శ్రద్ధ అవసరమయ్యే సెల్‌లను హైలైట్ చేయాలనుకుంటున్నారు. ఉదాహరణకు, ఫెయిల్స్, అధిక స్థాయి ఫ్రాగ్మెంటేషన్ మొదలైనవి. వాస్తవానికి, ఇది బేర్ SQLలో చేయవచ్చు, PRINTని ఉపయోగించి HTMLని రూపొందించడం మరియు జెంకిన్స్‌లో ఫైల్ రకాన్ని HTMLకి సెట్ చేయడం:

declare @body varchar(max), @chunk varchar(max)
set @body='<font face="Lucida Console" size="3">'
set @body=@body+'<b>Server name: '+@@servername+'</b><br>'
set @body=@body+'<br><br>'
set @body=@body+'<table><tr><th>Job</th><th>Last Run</th><th>Avg Duration, sec</th><th>Last Run, Sec</th><th>Last Status</th></tr>'
print @body

DECLARE tab CURSOR FOR SELECT '<tr><td>'+name+'</td><td>'+
  LastRun+'</td><td>'+
  convert(varchar,AvgDuration)+'</td><td>'+
  convert(varchar,LastDuration)+'</td><td>'+
    case when LastStatus<>'Succeeded' then '<font color="red">' else '' end+
      LastStatus+
      case when LastStatus<>'Succeeded' then '</font>' else '' end+
     +'</td><td>'
  from #j2
OPEN tab;  
FETCH NEXT FROM tab into @chunk
WHILE @@FETCH_STATUS = 0  
BEGIN
  print @chunk
  FETCH NEXT FROM tab into @chunk;  
END  
CLOSE tab;  
DEALLOCATE tab;
print '</table>'

నేను అలాంటి కోడ్ ఎందుకు వ్రాసాను?

జెంకిన్స్‌లో SQL సర్వర్ యొక్క ఆటోమేషన్: ఫలితాన్ని అందంగా అందిస్తుంది

కానీ మరింత అందమైన పరిష్కారం ఉంది. HTMLకి మార్చండి కణాలకు రంగు వేయనివ్వదు, కానీ వాస్తవం తర్వాత మనం దీన్ని చేయవచ్చు. ఉదాహరణకు, మేము 80 కంటే ఎక్కువ మరియు 90 కంటే ఎక్కువ ఫ్రాగ్మెంటేషన్ స్థాయిని కలిగి ఉన్న సెల్‌లను ఎంచుకోవాలనుకుంటున్నాము. శైలులను జోడిద్దాం:

<style>
.SQLmarkup-red { color: red; background-color: yellow; }
.SQLmarkup-yellow { color: black; background-color: #FFFFE0; }
.SQLmarkup-default { color: black; background-color: white; }
</style>

ప్రశ్నలోనే మేము డమ్మీ కాలమ్‌ని జోడిస్తాము వెంటనే ముందు మేము రంగు వేయాలనుకుంటున్న కాలమ్. కాలమ్ అని పిలవాలి SQLmarkup-ఏదో:

case  
  when ps.avg_fragmentation_in_percent>=90.0 then 'SQLmarkup-red'
  when ps.avg_fragmentation_in_percent>=80.0 then 'SQLmarkup-yellow'
  else 'SQLmarkup-default' 
  end as [SQLmarkup-1], 
ps.avg_fragmentation_in_percent, 

ఇప్పుడు, పవర్‌షెల్ ద్వారా రూపొందించబడిన HTMLను స్వీకరించిన తర్వాత, మేము హెడర్ నుండి డమ్మీ కాలమ్‌ను తీసివేస్తాము మరియు డేటా యొక్క బాడీలో మేము విలువను కాలమ్ నుండి శైలికి బదిలీ చేస్తాము. ఇది కేవలం రెండు ప్రత్యామ్నాయాలతో చేయబడుతుంది:

$html = $html `
  -replace "<th>SQLmarkup[^<]*</th>", "" `
  -replace "<td>SQLmarkup-(.+?)</td><td>",'<td class="SQLmarkup-$1">'

ఫలితంగా:
జెంకిన్స్‌లో SQL సర్వర్ యొక్క ఆటోమేషన్: ఫలితాన్ని అందంగా అందిస్తుంది

సొగసైనది కాదా? లేనప్పటికీ, ఈ రంగు నాకు ఏదో గుర్తుచేస్తుంది
జెంకిన్స్‌లో SQL సర్వర్ యొక్క ఆటోమేషన్: ఫలితాన్ని అందంగా అందిస్తుంది

మూలం: www.habr.com

ఒక వ్యాఖ్యను జోడించండి