|
| 1 | +unit uDatabaseStats; |
| 2 | + |
| 3 | +interface |
| 4 | + |
| 5 | +uses |
| 6 | + Classes, uPageAnalyzer, uDataPage, uTipPage, uBtreePage; |
| 7 | + |
| 8 | +type |
| 9 | + SChar = Shortint; |
| 10 | + SShort = Smallint; |
| 11 | + UShort = Word; |
| 12 | + SLong = Longint; |
| 13 | + ULong = LongWord; |
| 14 | + UChar = type Byte; |
| 15 | + |
| 16 | + TPageTypeStats = record |
| 17 | + HeaderPages: ULong; |
| 18 | + PipPages: ULong; |
| 19 | + TipPages: ULong; |
| 20 | + PointerPages: ULong; |
| 21 | + DataPages: ULong; |
| 22 | + IndexRootPages: ULong; |
| 23 | + IndexBtreePages: ULong; |
| 24 | + BlobPages: ULong; |
| 25 | + GeneratorPages: ULong; |
| 26 | + WalPages: ULong; |
| 27 | + UnknownPages: ULong; |
| 28 | + end; |
| 29 | + |
| 30 | + |
| 31 | + TTransactionStats = record |
| 32 | + Active: ULong; |
| 33 | + Limbo: ULong; |
| 34 | + Dead: ULong; |
| 35 | + Committed: ULong; |
| 36 | + Total: ULong; |
| 37 | + end; |
| 38 | + |
| 39 | + |
| 40 | + TRelationStats = record |
| 41 | + RelationID: ULong; |
| 42 | + PageCount: ULong; // Êîëè÷åñòâî Data Pages, ïðèíàäëåæàùèõ îòíîøåíèþ |
| 43 | + RecordCount: ULong; // Ïðèìåð: êîëè÷åñòâî çàïèñåé (òðåáóåò ïàðñèíãà ôðàãìåíòîâ) |
| 44 | + end; |
| 45 | + TRelationStatsArray = array of TRelationStats; |
| 46 | + |
| 47 | + TDatabaseStats = class |
| 48 | + private |
| 49 | + FPageAnalyzer: TPageAnalyzer; |
| 50 | + FPageStats: TPageTypeStats; |
| 51 | + FTransactionStats: TTransactionStats; |
| 52 | + FRelationStats: TRelationStatsArray; |
| 53 | + FStatsCalculated: Boolean; |
| 54 | + |
| 55 | + // Âíóòðåííÿÿ ôóíêöèÿ äëÿ ïîäñ÷åòà òèïîâ ñòðàíèö |
| 56 | + procedure CalculatePageStats; |
| 57 | + // Âíóòðåííÿÿ ôóíêöèÿ äëÿ ïîäñ÷åòà ñòàòèñòèêè ïî TIP |
| 58 | + procedure CalculateTransactionStats; |
| 59 | + // Âíóòðåííÿÿ ôóíêöèÿ äëÿ ïîäñ÷åòà ñòàòèñòèêè ïî îòíîøåíèÿì (óïðîùåííàÿ) |
| 60 | + procedure CalculateRelationStats; |
| 61 | + public |
| 62 | + // Êîíñòðóêòîð - ïðèíèìàåò TPageAnalyzer |
| 63 | + constructor Create(APageAnalyzer: TPageAnalyzer); |
| 64 | + // Äåñòðóêòîð |
| 65 | + destructor Destroy; override; |
| 66 | + |
| 67 | + // Âû÷èñëèòü âñþ ñòàòèñòèêó |
| 68 | + procedure CalculateStats; |
| 69 | + |
| 70 | + // Ïîëó÷èòü ñòàòèñòèêó ïî òèïàì ñòðàíèö |
| 71 | + property PageStats: TPageTypeStats read FPageStats; |
| 72 | + // Ïîëó÷èòü ñòàòèñòèêó ïî òðàíçàêöèÿì |
| 73 | + property TransactionStats: TTransactionStats read FTransactionStats; |
| 74 | + // Ïîëó÷èòü ñòàòèñòèêó ïî îòíîøåíèÿì |
| 75 | + property RelationStats: TRelationStatsArray read FRelationStats; |
| 76 | + // Ñòàòèñòèêà óæå âû÷èñëåíà? |
| 77 | + property StatsCalculated: Boolean read FStatsCalculated; |
| 78 | + end; |
| 79 | + |
| 80 | +implementation |
| 81 | + |
| 82 | +uses |
| 83 | + SysUtils; |
| 84 | + |
| 85 | +constructor TDatabaseStats.Create(APageAnalyzer: TPageAnalyzer); |
| 86 | +begin |
| 87 | + inherited Create; |
| 88 | + FPageAnalyzer := APageAnalyzer; |
| 89 | + FStatsCalculated := False; |
| 90 | +end; |
| 91 | + |
| 92 | +destructor TDatabaseStats.Destroy; |
| 93 | +begin |
| 94 | + inherited; |
| 95 | +end; |
| 96 | + |
| 97 | +procedure TDatabaseStats.CalculatePageStats; |
| 98 | +var |
| 99 | + PageNum: ULong; |
| 100 | + PageInfo: TPageInfo; |
| 101 | +begin |
| 102 | + |
| 103 | + FillChar(FPageStats, SizeOf(FPageStats), 0); |
| 104 | + |
| 105 | + for PageNum := 0 to FPageAnalyzer.GetLastPageNumber do |
| 106 | + begin |
| 107 | + PageInfo := FPageAnalyzer.GetPageInfo(PageNum); |
| 108 | + case PageInfo.Header.pag_type of |
| 109 | + pag_header: Inc(FPageStats.HeaderPages); |
| 110 | + pag_pip: Inc(FPageStats.PipPages); |
| 111 | + pag_tip: Inc(FPageStats.TipPages); |
| 112 | + pag_pointer: Inc(FPageStats.PointerPages); |
| 113 | + pag_data: Inc(FPageStats.DataPages); |
| 114 | + pag_index_root: Inc(FPageStats.IndexRootPages); |
| 115 | + pag_index_bt: Inc(FPageStats.IndexBtreePages); |
| 116 | + pag_blob: Inc(FPageStats.BlobPages); |
| 117 | + pag_generator: Inc(FPageStats.GeneratorPages); |
| 118 | + pag_wal: Inc(FPageStats.WalPages); |
| 119 | + else |
| 120 | + Inc(FPageStats.UnknownPages); |
| 121 | + end; |
| 122 | + end; |
| 123 | +end; |
| 124 | + |
| 125 | +procedure TDatabaseStats.CalculateTransactionStats; |
| 126 | +var |
| 127 | + PageNum: ULong; |
| 128 | + PageInfo: TPageInfo; |
| 129 | + TipAnalyzer: TTIPAnalyzer; |
| 130 | + TxInfoArray: TTransactionInfoArray; |
| 131 | + i: Integer; |
| 132 | +begin |
| 133 | + // Èíèöèàëèçèðóåì ñ÷åò÷èêè |
| 134 | + FillChar(FTransactionStats, SizeOf(FTransactionStats), 0); |
| 135 | + |
| 136 | + for PageNum := 0 to FPageAnalyzer.GetLastPageNumber do |
| 137 | + begin |
| 138 | + PageInfo := FPageAnalyzer.GetPageInfo(PageNum); |
| 139 | + if PageInfo.Header.pag_type = pag_tip then |
| 140 | + begin |
| 141 | + TipAnalyzer := TTIPAnalyzer.Create(PageInfo.Buffer, PageInfo.Size, PageInfo.Number); |
| 142 | + try |
| 143 | + TxInfoArray := TipAnalyzer.GetTransactionInfo(); // Ïîëó÷èòü âñå òðàíçàêöèè íà ñòðàíèöå |
| 144 | + for i := 0 to High(TxInfoArray) do |
| 145 | + begin |
| 146 | + case TxInfoArray[i].State of |
| 147 | + tsActive: Inc(FTransactionStats.Active); |
| 148 | + tsLimbo: Inc(FTransactionStats.Limbo); |
| 149 | + tsDead: Inc(FTransactionStats.Dead); |
| 150 | + tsCommitted: Inc(FTransactionStats.Committed); |
| 151 | + end; |
| 152 | + Inc(FTransactionStats.Total); |
| 153 | + end; |
| 154 | + finally |
| 155 | + TipAnalyzer.Free; |
| 156 | + end; |
| 157 | + end; |
| 158 | + end; |
| 159 | +end; |
| 160 | + |
| 161 | +procedure TDatabaseStats.CalculateRelationStats; |
| 162 | +var |
| 163 | + PageNum: ULong; |
| 164 | + PageInfo: TPageInfo; |
| 165 | + DataPageHeader: TData_Page_Fixed_Part; |
| 166 | + Fragments: TRecordFragmentsArray; |
| 167 | + FoundRelation: Boolean; |
| 168 | + i, j: Integer; |
| 169 | +begin |
| 170 | + SetLength(FRelationStats, 0); |
| 171 | + |
| 172 | + for PageNum := 0 to FPageAnalyzer.GetLastPageNumber do |
| 173 | + begin |
| 174 | + PageInfo := FPageAnalyzer.GetPageInfo(PageNum); |
| 175 | + if PageInfo.Header.pag_type = pag_data then |
| 176 | + begin |
| 177 | + if Length(PageInfo.Buffer) >= SizeOf(TData_Page_Fixed_Part) then |
| 178 | + begin |
| 179 | + DataPageHeader := TData_Page_Fixed_Part(PageInfo.Buffer[0]); |
| 180 | + var RelationID := DataPageHeader.dpg_relation; |
| 181 | + |
| 182 | + FoundRelation := False; |
| 183 | + for j := 0 to High(FRelationStats) do |
| 184 | + begin |
| 185 | + if FRelationStats[j].RelationID = RelationID then |
| 186 | + begin |
| 187 | + Inc(FRelationStats[j].PageCount); |
| 188 | + FoundRelation := True; |
| 189 | + Break; |
| 190 | + end; |
| 191 | + end; |
| 192 | + |
| 193 | + |
| 194 | + if not FoundRelation then |
| 195 | + begin |
| 196 | + SetLength(FRelationStats, Length(FRelationStats) + 1); |
| 197 | + FRelationStats[High(FRelationStats)].RelationID := RelationID; |
| 198 | + FRelationStats[High(FRelationStats)].PageCount := 1; |
| 199 | + FRelationStats[High(FRelationStats)].RecordCount := 0; |
| 200 | + end; |
| 201 | + |
| 202 | + |
| 203 | + Fragments := ExtractDataFragments(PageInfo.Buffer, PageInfo.Size); |
| 204 | + |
| 205 | + for j := 0 to High(FRelationStats) do |
| 206 | + begin |
| 207 | + if FRelationStats[j].RelationID = RelationID then |
| 208 | + begin |
| 209 | + Inc(FRelationStats[j].RecordCount, Length(Fragments)); |
| 210 | + Break; |
| 211 | + end; |
| 212 | + end; |
| 213 | + end; |
| 214 | + end; |
| 215 | + end; |
| 216 | +end; |
| 217 | + |
| 218 | +procedure TDatabaseStats.CalculateStats; |
| 219 | +begin |
| 220 | + CalculatePageStats; |
| 221 | + CalculateTransactionStats; |
| 222 | + CalculateRelationStats; |
| 223 | + FStatsCalculated := True; |
| 224 | +end; |
| 225 | + |
| 226 | +end. |
0 commit comments