testutil.go 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. // Copyright 2015 The etcd Authors
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. // Package testutil provides test utility functions.
  15. package testutil
  16. import (
  17. "net/url"
  18. "runtime"
  19. "testing"
  20. "time"
  21. )
  22. // WaitSchedule briefly sleeps in order to invoke the go scheduler.
  23. // TODO: improve this when we are able to know the schedule or status of target go-routine.
  24. func WaitSchedule() {
  25. time.Sleep(10 * time.Millisecond)
  26. }
  27. func MustNewURLs(t *testing.T, urls []string) []url.URL {
  28. if urls == nil {
  29. return nil
  30. }
  31. var us []url.URL
  32. for _, url := range urls {
  33. u := MustNewURL(t, url)
  34. us = append(us, *u)
  35. }
  36. return us
  37. }
  38. func MustNewURL(t *testing.T, s string) *url.URL {
  39. u, err := url.Parse(s)
  40. if err != nil {
  41. t.Fatalf("parse %v error: %v", s, err)
  42. }
  43. return u
  44. }
  45. // FatalStack helps to fatal the test and print out the stacks of all running goroutines.
  46. func FatalStack(t *testing.T, s string) {
  47. stackTrace := make([]byte, 1024*1024)
  48. n := runtime.Stack(stackTrace, true)
  49. t.Error(string(stackTrace[:n]))
  50. t.Fatalf(s)
  51. }
  52. // ConditionFunc returns true when a condition is met.
  53. type ConditionFunc func() (bool, error)
  54. // Poll calls a condition function repeatedly on a polling interval until it returns true, returns an error
  55. // or the timeout is reached. If the condition function returns true or an error before the timeout, Poll
  56. // immediately returns with the true value or the error. If the timeout is exceeded, Poll returns false.
  57. func Poll(interval time.Duration, timeout time.Duration, condition ConditionFunc) (bool, error) {
  58. timeoutCh := time.After(timeout)
  59. ticker := time.NewTicker(interval)
  60. defer ticker.Stop()
  61. for {
  62. select {
  63. case <-timeoutCh:
  64. return false, nil
  65. case <-ticker.C:
  66. success, err := condition()
  67. if err != nil {
  68. return false, err
  69. }
  70. if success {
  71. return true, nil
  72. }
  73. }
  74. }
  75. }