go.temporal.io/server@v1.23.0/common/persistence/sql/sqlplugin/tests/history_execution_timer.go (about) 1 // The MIT License 2 // 3 // Copyright (c) 2020 Temporal Technologies Inc. All rights reserved. 4 // 5 // Copyright (c) 2020 Uber Technologies, Inc. 6 // 7 // Permission is hereby granted, free of charge, to any person obtaining a copy 8 // of this software and associated documentation files (the "Software"), to deal 9 // in the Software without restriction, including without limitation the rights 10 // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 11 // copies of the Software, and to permit persons to whom the Software is 12 // furnished to do so, subject to the following conditions: 13 // 14 // The above copyright notice and this permission notice shall be included in 15 // all copies or substantial portions of the Software. 16 // 17 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 18 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 19 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 20 // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 21 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 22 // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN 23 // THE SOFTWARE. 24 25 package tests 26 27 import ( 28 "math/rand" 29 "testing" 30 31 "github.com/stretchr/testify/require" 32 "github.com/stretchr/testify/suite" 33 34 "go.temporal.io/server/common/persistence/sql/sqlplugin" 35 "go.temporal.io/server/common/primitives" 36 "go.temporal.io/server/common/shuffle" 37 ) 38 39 type ( 40 historyExecutionTimerSuite struct { 41 suite.Suite 42 *require.Assertions 43 44 store sqlplugin.HistoryExecutionTimer 45 } 46 ) 47 48 const ( 49 testHistoryExecutionTimerID = "random history timer ID" 50 testHistoryExecutionTimerEncoding = "random encoding" 51 ) 52 53 var ( 54 testHistoryExecutionTimerData = []byte("random history execution timer data") 55 ) 56 57 func NewHistoryExecutionTimerSuite( 58 t *testing.T, 59 store sqlplugin.HistoryExecutionTimer, 60 ) *historyExecutionTimerSuite { 61 return &historyExecutionTimerSuite{ 62 Assertions: require.New(t), 63 store: store, 64 } 65 } 66 67 func (s *historyExecutionTimerSuite) SetupSuite() { 68 69 } 70 71 func (s *historyExecutionTimerSuite) TearDownSuite() { 72 73 } 74 75 func (s *historyExecutionTimerSuite) SetupTest() { 76 s.Assertions = require.New(s.T()) 77 } 78 79 func (s *historyExecutionTimerSuite) TearDownTest() { 80 81 } 82 83 func (s *historyExecutionTimerSuite) TestReplace_Single() { 84 shardID := rand.Int31() 85 namespaceID := primitives.NewUUID() 86 workflowID := shuffle.String(testHistoryExecutionWorkflowID) 87 runID := primitives.NewUUID() 88 timerID := shuffle.String(testHistoryExecutionTimerID) 89 90 timer := s.newRandomExecutionTimerRow(shardID, namespaceID, workflowID, runID, timerID) 91 result, err := s.store.ReplaceIntoTimerInfoMaps(newExecutionContext(), []sqlplugin.TimerInfoMapsRow{timer}) 92 s.NoError(err) 93 rowsAffected, err := result.RowsAffected() 94 s.NoError(err) 95 s.Equal(1, int(rowsAffected)) 96 } 97 98 func (s *historyExecutionTimerSuite) TestReplace_Multiple() { 99 shardID := rand.Int31() 100 namespaceID := primitives.NewUUID() 101 workflowID := shuffle.String(testHistoryExecutionWorkflowID) 102 runID := primitives.NewUUID() 103 104 timer1 := s.newRandomExecutionTimerRow(shardID, namespaceID, workflowID, runID, shuffle.String(testHistoryExecutionTimerID)) 105 timer2 := s.newRandomExecutionTimerRow(shardID, namespaceID, workflowID, runID, shuffle.String(testHistoryExecutionTimerID)) 106 result, err := s.store.ReplaceIntoTimerInfoMaps(newExecutionContext(), []sqlplugin.TimerInfoMapsRow{timer1, timer2}) 107 s.NoError(err) 108 rowsAffected, err := result.RowsAffected() 109 s.NoError(err) 110 s.Equal(2, int(rowsAffected)) 111 } 112 113 func (s *historyExecutionTimerSuite) TestReplaceSelect_Single() { 114 shardID := rand.Int31() 115 namespaceID := primitives.NewUUID() 116 workflowID := shuffle.String(testHistoryExecutionWorkflowID) 117 runID := primitives.NewUUID() 118 timerID := shuffle.String(testHistoryExecutionTimerID) 119 120 timer := s.newRandomExecutionTimerRow(shardID, namespaceID, workflowID, runID, timerID) 121 result, err := s.store.ReplaceIntoTimerInfoMaps(newExecutionContext(), []sqlplugin.TimerInfoMapsRow{timer}) 122 s.NoError(err) 123 rowsAffected, err := result.RowsAffected() 124 s.NoError(err) 125 s.Equal(1, int(rowsAffected)) 126 127 selectFilter := sqlplugin.TimerInfoMapsAllFilter{ 128 ShardID: shardID, 129 NamespaceID: namespaceID, 130 WorkflowID: workflowID, 131 RunID: runID, 132 } 133 rows, err := s.store.SelectAllFromTimerInfoMaps(newExecutionContext(), selectFilter) 134 s.NoError(err) 135 rowMap := map[string]sqlplugin.TimerInfoMapsRow{} 136 for _, timer := range rows { 137 rowMap[timer.TimerID] = timer 138 } 139 s.Equal(map[string]sqlplugin.TimerInfoMapsRow{ 140 timer.TimerID: timer, 141 }, rowMap) 142 } 143 144 func (s *historyExecutionTimerSuite) TestReplaceSelect_Multiple() { 145 numTimers := 20 146 147 shardID := rand.Int31() 148 namespaceID := primitives.NewUUID() 149 workflowID := shuffle.String(testHistoryExecutionWorkflowID) 150 runID := primitives.NewUUID() 151 152 var timers []sqlplugin.TimerInfoMapsRow 153 for i := 0; i < numTimers; i++ { 154 timer := s.newRandomExecutionTimerRow(shardID, namespaceID, workflowID, runID, shuffle.String(testHistoryExecutionTimerID)) 155 timers = append(timers, timer) 156 } 157 result, err := s.store.ReplaceIntoTimerInfoMaps(newExecutionContext(), timers) 158 s.NoError(err) 159 rowsAffected, err := result.RowsAffected() 160 s.NoError(err) 161 s.Equal(numTimers, int(rowsAffected)) 162 163 selectFilter := sqlplugin.TimerInfoMapsAllFilter{ 164 ShardID: shardID, 165 NamespaceID: namespaceID, 166 WorkflowID: workflowID, 167 RunID: runID, 168 } 169 rows, err := s.store.SelectAllFromTimerInfoMaps(newExecutionContext(), selectFilter) 170 s.NoError(err) 171 timerMap := map[string]sqlplugin.TimerInfoMapsRow{} 172 for _, timer := range timers { 173 timerMap[timer.TimerID] = timer 174 } 175 rowMap := map[string]sqlplugin.TimerInfoMapsRow{} 176 for _, timer := range rows { 177 rowMap[timer.TimerID] = timer 178 } 179 s.Equal(timerMap, rowMap) 180 } 181 182 func (s *historyExecutionTimerSuite) TestDeleteSelect_Single() { 183 shardID := rand.Int31() 184 namespaceID := primitives.NewUUID() 185 workflowID := shuffle.String(testHistoryExecutionWorkflowID) 186 runID := primitives.NewUUID() 187 timerID := shuffle.String(testHistoryExecutionTimerID) 188 189 deletFilter := sqlplugin.TimerInfoMapsFilter{ 190 ShardID: shardID, 191 NamespaceID: namespaceID, 192 WorkflowID: workflowID, 193 RunID: runID, 194 TimerIDs: []string{timerID}, 195 } 196 result, err := s.store.DeleteFromTimerInfoMaps(newExecutionContext(), deletFilter) 197 s.NoError(err) 198 rowsAffected, err := result.RowsAffected() 199 s.NoError(err) 200 s.Equal(0, int(rowsAffected)) 201 202 selectFilter := sqlplugin.TimerInfoMapsAllFilter{ 203 ShardID: shardID, 204 NamespaceID: namespaceID, 205 WorkflowID: workflowID, 206 RunID: runID, 207 } 208 rows, err := s.store.SelectAllFromTimerInfoMaps(newExecutionContext(), selectFilter) 209 s.NoError(err) 210 s.Equal([]sqlplugin.TimerInfoMapsRow(nil), rows) 211 } 212 213 func (s *historyExecutionTimerSuite) TestDeleteSelect_Multiple() { 214 shardID := rand.Int31() 215 namespaceID := primitives.NewUUID() 216 workflowID := shuffle.String(testHistoryExecutionWorkflowID) 217 runID := primitives.NewUUID() 218 219 deleteFilter := sqlplugin.TimerInfoMapsFilter{ 220 ShardID: shardID, 221 NamespaceID: namespaceID, 222 WorkflowID: workflowID, 223 RunID: runID, 224 TimerIDs: []string{shuffle.String(testHistoryExecutionTimerID), shuffle.String(testHistoryExecutionTimerID)}, 225 } 226 result, err := s.store.DeleteFromTimerInfoMaps(newExecutionContext(), deleteFilter) 227 s.NoError(err) 228 rowsAffected, err := result.RowsAffected() 229 s.NoError(err) 230 s.Equal(0, int(rowsAffected)) 231 232 selectFilter := sqlplugin.TimerInfoMapsAllFilter{ 233 ShardID: shardID, 234 NamespaceID: namespaceID, 235 WorkflowID: workflowID, 236 RunID: runID, 237 } 238 rows, err := s.store.SelectAllFromTimerInfoMaps(newExecutionContext(), selectFilter) 239 s.NoError(err) 240 s.Equal([]sqlplugin.TimerInfoMapsRow(nil), rows) 241 } 242 243 func (s *historyExecutionTimerSuite) TestDeleteSelect_All() { 244 shardID := rand.Int31() 245 namespaceID := primitives.NewUUID() 246 workflowID := shuffle.String(testHistoryExecutionWorkflowID) 247 runID := primitives.NewUUID() 248 249 deleteFilter := sqlplugin.TimerInfoMapsAllFilter{ 250 ShardID: shardID, 251 NamespaceID: namespaceID, 252 WorkflowID: workflowID, 253 RunID: runID, 254 } 255 result, err := s.store.DeleteAllFromTimerInfoMaps(newExecutionContext(), deleteFilter) 256 s.NoError(err) 257 rowsAffected, err := result.RowsAffected() 258 s.NoError(err) 259 s.Equal(0, int(rowsAffected)) 260 261 selectFilter := sqlplugin.TimerInfoMapsAllFilter{ 262 ShardID: shardID, 263 NamespaceID: namespaceID, 264 WorkflowID: workflowID, 265 RunID: runID, 266 } 267 rows, err := s.store.SelectAllFromTimerInfoMaps(newExecutionContext(), selectFilter) 268 s.NoError(err) 269 s.Equal([]sqlplugin.TimerInfoMapsRow(nil), rows) 270 } 271 272 func (s *historyExecutionTimerSuite) TestReplaceDeleteSelect_Single() { 273 shardID := rand.Int31() 274 namespaceID := primitives.NewUUID() 275 workflowID := shuffle.String(testHistoryExecutionWorkflowID) 276 runID := primitives.NewUUID() 277 timerID := shuffle.String(testHistoryExecutionTimerID) 278 279 timer := s.newRandomExecutionTimerRow(shardID, namespaceID, workflowID, runID, timerID) 280 result, err := s.store.ReplaceIntoTimerInfoMaps(newExecutionContext(), []sqlplugin.TimerInfoMapsRow{timer}) 281 s.NoError(err) 282 rowsAffected, err := result.RowsAffected() 283 s.NoError(err) 284 s.Equal(1, int(rowsAffected)) 285 286 deleteFilter := sqlplugin.TimerInfoMapsFilter{ 287 ShardID: shardID, 288 NamespaceID: namespaceID, 289 WorkflowID: workflowID, 290 RunID: runID, 291 TimerIDs: []string{timerID}, 292 } 293 result, err = s.store.DeleteFromTimerInfoMaps(newExecutionContext(), deleteFilter) 294 s.NoError(err) 295 rowsAffected, err = result.RowsAffected() 296 s.NoError(err) 297 s.Equal(1, int(rowsAffected)) 298 299 selectFilter := sqlplugin.TimerInfoMapsAllFilter{ 300 ShardID: shardID, 301 NamespaceID: namespaceID, 302 WorkflowID: workflowID, 303 RunID: runID, 304 } 305 rows, err := s.store.SelectAllFromTimerInfoMaps(newExecutionContext(), selectFilter) 306 s.NoError(err) 307 s.Equal([]sqlplugin.TimerInfoMapsRow(nil), rows) 308 } 309 310 func (s *historyExecutionTimerSuite) TestReplaceDeleteSelect_Multiple() { 311 numTimers := 20 312 313 shardID := rand.Int31() 314 namespaceID := primitives.NewUUID() 315 workflowID := shuffle.String(testHistoryExecutionWorkflowID) 316 runID := primitives.NewUUID() 317 318 var timers []sqlplugin.TimerInfoMapsRow 319 var timerIDs []string 320 for i := 0; i < numTimers; i++ { 321 timerID := shuffle.String(testHistoryExecutionTimerID) 322 timer := s.newRandomExecutionTimerRow(shardID, namespaceID, workflowID, runID, timerID) 323 timerIDs = append(timerIDs, timerID) 324 timers = append(timers, timer) 325 } 326 result, err := s.store.ReplaceIntoTimerInfoMaps(newExecutionContext(), timers) 327 s.NoError(err) 328 rowsAffected, err := result.RowsAffected() 329 s.NoError(err) 330 s.Equal(numTimers, int(rowsAffected)) 331 332 deleteFilter := sqlplugin.TimerInfoMapsFilter{ 333 ShardID: shardID, 334 NamespaceID: namespaceID, 335 WorkflowID: workflowID, 336 RunID: runID, 337 TimerIDs: timerIDs, 338 } 339 result, err = s.store.DeleteFromTimerInfoMaps(newExecutionContext(), deleteFilter) 340 s.NoError(err) 341 rowsAffected, err = result.RowsAffected() 342 s.NoError(err) 343 s.Equal(numTimers, int(rowsAffected)) 344 345 selectFilter := sqlplugin.TimerInfoMapsAllFilter{ 346 ShardID: shardID, 347 NamespaceID: namespaceID, 348 WorkflowID: workflowID, 349 RunID: runID, 350 } 351 rows, err := s.store.SelectAllFromTimerInfoMaps(newExecutionContext(), selectFilter) 352 s.NoError(err) 353 s.Equal([]sqlplugin.TimerInfoMapsRow(nil), rows) 354 } 355 356 func (s *historyExecutionTimerSuite) TestReplaceDeleteSelect_All() { 357 numTimers := 20 358 359 shardID := rand.Int31() 360 namespaceID := primitives.NewUUID() 361 workflowID := shuffle.String(testHistoryExecutionWorkflowID) 362 runID := primitives.NewUUID() 363 364 var timers []sqlplugin.TimerInfoMapsRow 365 for i := 0; i < numTimers; i++ { 366 timer := s.newRandomExecutionTimerRow(shardID, namespaceID, workflowID, runID, shuffle.String(testHistoryExecutionTimerID)) 367 timers = append(timers, timer) 368 } 369 result, err := s.store.ReplaceIntoTimerInfoMaps(newExecutionContext(), timers) 370 s.NoError(err) 371 rowsAffected, err := result.RowsAffected() 372 s.NoError(err) 373 s.Equal(numTimers, int(rowsAffected)) 374 375 deleteFilter := sqlplugin.TimerInfoMapsAllFilter{ 376 ShardID: shardID, 377 NamespaceID: namespaceID, 378 WorkflowID: workflowID, 379 RunID: runID, 380 } 381 result, err = s.store.DeleteAllFromTimerInfoMaps(newExecutionContext(), deleteFilter) 382 s.NoError(err) 383 rowsAffected, err = result.RowsAffected() 384 s.NoError(err) 385 s.Equal(numTimers, int(rowsAffected)) 386 387 selectFilter := sqlplugin.TimerInfoMapsAllFilter{ 388 ShardID: shardID, 389 NamespaceID: namespaceID, 390 WorkflowID: workflowID, 391 RunID: runID, 392 } 393 rows, err := s.store.SelectAllFromTimerInfoMaps(newExecutionContext(), selectFilter) 394 s.NoError(err) 395 s.Equal([]sqlplugin.TimerInfoMapsRow(nil), rows) 396 } 397 398 func (s *historyExecutionTimerSuite) newRandomExecutionTimerRow( 399 shardID int32, 400 namespaceID primitives.UUID, 401 workflowID string, 402 runID primitives.UUID, 403 timerID string, 404 ) sqlplugin.TimerInfoMapsRow { 405 return sqlplugin.TimerInfoMapsRow{ 406 ShardID: shardID, 407 NamespaceID: namespaceID, 408 WorkflowID: workflowID, 409 RunID: runID, 410 TimerID: timerID, 411 Data: shuffle.Bytes(testHistoryExecutionTimerData), 412 DataEncoding: testHistoryExecutionTimerEncoding, 413 } 414 }