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  }