Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[TT-13723] Update to Go 1.23 #6812

Merged
merged 11 commits into from
Jan 9, 2025
Merged

[TT-13723] Update to Go 1.23 #6812

merged 11 commits into from
Jan 9, 2025

Conversation

titpetric
Copy link
Contributor

@titpetric titpetric commented Dec 23, 2024

User description

TT-13723
Summary Update to Go 1.23
Type Story Story
Status In Dev
Points N/A
Labels -

https://tyktech.atlassian.net/browse/TT-13723

It seems some tests detect goroutine leaks now. The detected goroutines leaked have been listed in the ignores of a debug2.Record; both goroutine leak tests detect goroutines in background reliably. Both are flaky otherwise, this passes a -count=100 run, with and without -race.


PR Type

Enhancement, Tests, Configuration changes


Description

  • Introduced debug2.Record to enhance goroutine state tracking and comparison.
  • Improved goroutine leak detection tests using debug2.Record.
  • Added unit and benchmark tests for debug2.Record.
  • Updated CI workflows to use Go 1.23.x.
  • Simplified Dockerfile by switching to Go 1.23-bullseye base image and optimizing build steps.
  • Updated plugin compiler and release workflows to support Go 1.23.
  • Enhanced Taskfile to dynamically use the Go version from go.mod.
  • Bumped Go version in go.mod to 1.23.4.

Changes walkthrough 📝

Relevant files
Tests
2 files
gateway_test.go
Improved goroutine leak detection in tests.                           

gateway/gateway_test.go

  • Enhanced goroutine leak tests with debug2.Record for better
    reliability.
  • Introduced newRecord helper function to manage ignored goroutines.
  • Updated assertions to use debug2.Record for goroutine count
    validation.
  • +37/-15 
    goroutine_test.go
    Added unit and benchmark tests for `debug2.Record`.           

    internal/debug2/goroutine_test.go

  • Added unit tests for debug2.Record to validate goroutine tracking.
  • Included benchmark tests for performance evaluation.
  • Verified goroutine cleanup using Since method.
  • +103/-0 
    Enhancement
    4 files
    goroutine.go
    Introduced `debug2.Record` for goroutine state tracking. 

    internal/debug2/goroutine.go

  • Added debug2.Record to capture and compare goroutine states.
  • Implemented methods for parsing, counting, and filtering goroutines.
  • Introduced functionality to ignore specific goroutines during
    comparison.
  • +124/-0 
    Dockerfile
    Simplified Dockerfile with Go 1.23 base image.                     

    Dockerfile

  • Simplified Dockerfile by using Go 1.23-bullseye base image.
  • Removed redundant Python installation steps.
  • Optimized build process with caching for Go modules.
  • +11/-51 
    Taskfile.yml
    Enhanced Taskfile to support dynamic Go version.                 

    Taskfile.yml

    • Added dynamic Go version argument for Docker builds.
    +1/-1     
    go.mod
    Bumped Go version in module to 1.23.4.                                     

    go.mod

    • Updated Go version requirement to 1.23.4.
    +1/-1     
    Configuration changes
    4 files
    ci-tests.yml
    Updated CI workflow to use Go 1.23.x.                                       

    .github/workflows/ci-tests.yml

    • Updated Go version in CI matrix to 1.23.x.
    +1/-1     
    plugin-compiler-build.yml
    Updated plugin compiler workflow for Go 1.23.                       

    .github/workflows/plugin-compiler-build.yml

  • Updated base image to use Go 1.23-bullseye.
  • Fixed BASE_IMAGE argument in Docker build steps.
  • +3/-3     
    release.yml
    Updated release workflow to support Go 1.23.                         

    .github/workflows/release.yml

  • Updated Go version in release workflow to 1.23-bullseye.
  • Adjusted Docker build conditions for the new Go version.
  • +11/-11 
    Dockerfile
    Updated plugin compiler Dockerfile for Go 1.23.                   

    ci/images/plugin-compiler/Dockerfile

    • Updated base image to Go 1.23-bullseye.
    +1/-1     

    💡 PR-Agent usage: Comment /help "your question" on any pull request to receive relevant information

    @buger
    Copy link
    Member

    buger commented Dec 23, 2024

    I'm a bot and I 👍 this PR title. 🤖

    Copy link
    Contributor

    github-actions bot commented Dec 23, 2024

    API Changes

    --- prev.txt	2025-01-09 08:28:42.997404757 +0000
    +++ current.txt	2025-01-09 08:28:38.278382577 +0000
    @@ -12495,6 +12495,8 @@
     
     package regression // import "github.com/TykTechnologies/tyk/tests/regression"
     
    +# Package: ./tests/system
    +
     # Package: ./trace
     
     package trace // import "github.com/TykTechnologies/tyk/trace"

    Copy link
    Contributor

    PR Reviewer Guide 🔍

    Here are some key observations to aid the review process:

    ⏱️ Estimated effort to review: 4 🔵🔵🔵🔵⚪
    🧪 PR contains tests
    🔒 No security concerns identified
    ⚡ Recommended focus areas for review

    Possible Flaky Test

    The new goroutine leak tests rely on time.Sleep and runtime.GC() for synchronization, which might lead to flaky behavior in different environments. Consider using more robust synchronization mechanisms.

    func TestReloadGoroutineLeakWithTest(t *testing.T) {
    	newRecord := func() *debug2.Record {
    		result := debug2.NewRecord()
    		result.SetIgnores([]string{
    			"runtime/pprof.writeRuntimeProfile",
    			"/root/go/pkg/mod/github.com/!tyk!technologies/[email protected]/memorycache/cache.go:69",
    			"/root/go/pkg/mod/github.com/pmylund/[email protected]+incompatible/cache.go:1079",
    			"/root/tyk/tyk/gateway/distributed_rate_limiter.go:31",
    			"/root/tyk/tyk/gateway/redis_signals.go:68",
    		})
    
    		return result
    	}
    
    	before := newRecord()
    
    	ts := StartTest(nil)
    	ts.Close()
    
    	time.Sleep(100 * time.Millisecond)
    	runtime.GC()
    
    	final := newRecord().Since(before)
    	assert.Equal(t, 0, final.Count(), "final count not zero: %s", final)
    }
    
    func TestReloadGoroutineLeakWithCircuitBreaker(t *testing.T) {
    	ts := StartTest(nil)
    	t.Cleanup(ts.Close)
    
    	newRecord := func() *debug2.Record {
    		result := debug2.NewRecord()
    		result.SetIgnores([]string{
    			"runtime/pprof.writeRuntimeProfile",
    			"/root/tyk/tyk/gateway/reverse_proxy.go:223",
    			"/root/tyk/tyk/gateway/api_definition.go:1025",
    			"/root/tyk/tyk/gateway/distributed_rate_limiter.go:31",
    			"/root/go/pkg/mod/github.com/pmylund/[email protected]+incompatible/cache.go:1079",
    			"/root/go/pkg/mod/github.com/!tyk!technologies/[email protected]+incompatible/circuitbreaker.go:202",
    		})
    
    		return result
    	}
    
    	globalConf := ts.Gw.GetConfig()
    	globalConf.EnableJSVM = false
    	ts.Gw.SetConfig(globalConf)
    
    	stage1 := newRecord()
    
    	specs := ts.Gw.BuildAndLoadAPI(func(spec *APISpec) {
    		spec.Proxy.ListenPath = "/"
    		UpdateAPIVersion(spec, "v1", func(version *apidef.VersionInfo) {
    			version.ExtendedPaths = apidef.ExtendedPathsSet{
    				CircuitBreaker: []apidef.CircuitBreakerMeta{
    					{
    						Path:                 "/",
    						Method:               http.MethodGet,
    						ThresholdPercent:     0.5,
    						Samples:              5,
    						ReturnToServiceAfter: 10,
    					},
    				},
    			}
    		})
    	})
    
    	ts.Gw.LoadAPI(specs...) // just doing globalGateway.DoReload() doesn't load anything as BuildAndLoadAPI cleans up folder with API specs
    
    	time.Sleep(100 * time.Millisecond)
    	runtime.GC()
    
    	final := newRecord().Since(stage1)
    	assert.Equal(t, 0, final.Count(), "final count not zero: %s", final)
    }
    Performance Concern

    The parseGoroutines function processes goroutine dumps line by line and performs multiple string operations, which might become a performance bottleneck for large dumps. Consider optimizing this logic.

    func (r *Record) parseGoroutines() map[string][]string {
    	goroutines := make(map[string][]string)
    	var currentHeader string
    	var currentStack []string
    	toDelete := []string{}
    	lines := strings.Split(r.buffer.String(), "\n")
    
    	for _, line := range lines {
    		var skip bool
    		for _, ign := range r.ignores {
    			if strings.Contains(line, ign) {
    				skip = true
    				break
    			}
    		}
    
    		if skip {
    			toDelete = append(toDelete, currentHeader)
    		}
    
    		if headerMatchRe.MatchString(line) {
    			// Save the previous goroutine and reset
    			if currentHeader != "" {
    				goroutines[currentHeader] = currentStack
    			}
    			currentHeader = line
    			currentStack = []string{line}
    		} else if currentHeader != "" {
    			// Add stack trace lines to the current goroutine
    			currentStack = append(currentStack, line)
    		}
    	}
    
    	// Save the last goroutine
    	if currentHeader != "" {
    		goroutines[currentHeader] = currentStack
    	}
    
    	for _, key := range toDelete {
    		delete(goroutines, key)
    	}
    
    	return goroutines

    Copy link
    Contributor

    PR Code Suggestions ✨

    Explore these optional code suggestions:

    CategorySuggestion                                                                                                                                    Score
    General
    Add a maximum iteration limit to prevent infinite loops during goroutine cleanup checks

    Add a timeout or fail-safe mechanism in the loop waiting for goroutines to finish to
    prevent infinite loops in case of unexpected behavior.

    internal/debug2/goroutine_test.go [40-57]

    -for {
    +for i := 0; i < 50; i++ { // Limit iterations to prevent infinite loop
         time.Sleep(100 * time.Millisecond)
         runtime.GC()
         time.Sleep(10 * time.Millisecond)
         ...
         if remainingGoroutines.Count() == 0 {
             break
         }
     }
    Suggestion importance[1-10]: 8

    Why: Adding a maximum iteration limit is a significant improvement to prevent potential infinite loops during goroutine cleanup checks, enhancing reliability and robustness of the test.

    8
    Remove unnecessary build dependencies to reduce the final Docker image size

    Add a cleanup step to remove unnecessary build dependencies after the build process
    to reduce the final image size.

    Dockerfile [6]

    -RUN apt update && apt install build-essential zlib1g-dev libncurses5-dev libgdbm-dev libnss3-dev libssl-dev libsqlite3-dev libreadline-dev libffi-dev curl wget libbz2-dev -y
    +RUN apt update && apt install build-essential zlib1g-dev libncurses5-dev libgdbm-dev libnss3-dev libssl-dev libsqlite3-dev libreadline-dev libffi-dev curl wget libbz2-dev -y && \
    +    apt-get remove --purge -y build-essential && apt-get autoremove -y && apt-get clean
    Suggestion importance[1-10]: 7

    Why: Removing unnecessary build dependencies is a good practice to reduce the final Docker image size, improving efficiency and security. This suggestion aligns well with the PR's context and goals.

    7
    Clear the toDelete slice after use to prevent potential memory leaks or unintended side effects

    Ensure that the toDelete slice is cleared after processing to avoid potential memory
    leaks or unintended behavior in subsequent calls to parseGoroutines.

    internal/debug2/goroutine.go [44-79]

     toDelete := []string{}
     ...
     for _, key := range toDelete {
         delete(goroutines, key)
     }
    +toDelete = nil
    Suggestion importance[1-10]: 3

    Why: Clearing the toDelete slice after use is a minor improvement for memory management, but it is not critical since Go's garbage collector would handle it. The impact on functionality is minimal.

    3

    Copy link
    Contributor

    @jeffy-mathew jeffy-mathew left a comment

    Choose a reason for hiding this comment

    The reason will be displayed to describe this comment to others. Learn more.

    LGTM.
    Regarding the go routine leak tests - they were already flaky and probably go1.23 reports it better. Since this is not something that got introduced in go1.23, I don't feel like this must be a blocker for go1.23 upgrade.

    Copy link

    sonarqubecloud bot commented Jan 9, 2025

    Quality Gate Failed Quality Gate failed

    Failed conditions
    1 Security Hotspot
    C Reliability Rating on New Code (required ≥ A)

    See analysis details on SonarQube Cloud

    Catch issues before they fail your Quality Gate with our IDE extension SonarQube for IDE

    @titpetric titpetric merged commit daf76a8 into master Jan 9, 2025
    39 of 41 checks passed
    @titpetric titpetric deleted the bump/go-1.23 branch January 9, 2025 10:47
    titpetric added a commit that referenced this pull request Jan 9, 2025
    <details open>
    <summary><a href="https://tyktech.atlassian.net/browse/TT-13723"
    title="TT-13723" target="_blank">TT-13723</a></summary>
      <br />
      <table>
        <tr>
          <th>Summary</th>
          <td>Update to Go 1.23</td>
        </tr>
        <tr>
          <th>Type</th>
          <td>
    <img alt="Story"
    src="https://tyktech.atlassian.net/rest/api/2/universal_avatar/view/type/issuetype/avatar/10315?size=medium"
    />
            Story
          </td>
        </tr>
        <tr>
          <th>Status</th>
          <td>In Dev</td>
        </tr>
        <tr>
          <th>Points</th>
          <td>N/A</td>
        </tr>
        <tr>
          <th>Labels</th>
          <td>-</td>
        </tr>
      </table>
    </details>
    <!--
      do not remove this marker as it will break jira-lint's functionality.
      added_by_jira_lint
    -->
    
    ---
    
    https://tyktech.atlassian.net/browse/TT-13723
    
    It seems some tests detect goroutine leaks now. The detected goroutines
    leaked have been listed in the ignores of a debug2.Record; both
    goroutine leak tests detect goroutines in background reliably. Both are
    flaky otherwise, this passes a -count=100 run, with and without -race.
    
    ___
    
    Enhancement, Tests, Configuration changes
    
    ___
    
    - Introduced `debug2.Record` to enhance goroutine state tracking and
    comparison.
    - Improved goroutine leak detection tests using `debug2.Record`.
    - Added unit and benchmark tests for `debug2.Record`.
    - Updated CI workflows to use Go 1.23.x.
    - Simplified Dockerfile by switching to Go 1.23-bullseye base image and
    optimizing build steps.
    - Updated plugin compiler and release workflows to support Go 1.23.
    - Enhanced Taskfile to dynamically use the Go version from `go.mod`.
    - Bumped Go version in `go.mod` to 1.23.4.
    
    ___
    
    <table><thead><tr><th></th><th align="left">Relevant
    files</th></tr></thead><tbody><tr><td><strong>Tests</strong></td><td><details><summary>2
    files</summary><table>
    <tr>
      <td>
        <details>
    <summary><strong>gateway_test.go</strong><dd><code>Improved goroutine
    leak detection in tests.</code>&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;
    &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; </dd></summary>
    <hr>
    
    gateway/gateway_test.go
    
    <li>Enhanced goroutine leak tests with <code>debug2.Record</code> for
    better <br>reliability.<br> <li> Introduced <code>newRecord</code>
    helper function to manage ignored goroutines.<br> <li> Updated
    assertions to use <code>debug2.Record</code> for goroutine count
    <br>validation.<br>
    
    </details>
    
      </td>
    <td><a
    href="https://github.com/TykTechnologies/tyk/pull/6812/files#diff-d34c7069ce5e81d45082b19eb3e869ee1a086e185dcd6630e75e3ed0d368b546">+37/-15</a>&nbsp;
    </td>
    
    </tr>
    
    <tr>
      <td>
        <details>
    <summary><strong>goroutine_test.go</strong><dd><code>Added unit and
    benchmark tests for `debug2.Record`.</code>&nbsp; &nbsp; &nbsp; &nbsp;
    &nbsp; &nbsp; </dd></summary>
    <hr>
    
    internal/debug2/goroutine_test.go
    
    <li>Added unit tests for <code>debug2.Record</code> to validate
    goroutine tracking.<br> <li> Included benchmark tests for performance
    evaluation.<br> <li> Verified goroutine cleanup using <code>Since</code>
    method.<br>
    
    </details>
    
      </td>
    <td><a
    href="https://github.com/TykTechnologies/tyk/pull/6812/files#diff-763a0643c49a4bbfa70fbf12088045a74be7fc5f4789d09f14a9298d0b65c226">+103/-0</a>&nbsp;
    </td>
    
    </tr>
    
    </table></details></td></tr><tr><td><strong>Enhancement</strong></td><td><details><summary>4
    files</summary><table>
    <tr>
      <td>
        <details>
    <summary><strong>goroutine.go</strong><dd><code>Introduced
    `debug2.Record` for goroutine state tracking.</code>&nbsp;
    </dd></summary>
    <hr>
    
    internal/debug2/goroutine.go
    
    <li>Added <code>debug2.Record</code> to capture and compare goroutine
    states.<br> <li> Implemented methods for parsing, counting, and
    filtering goroutines.<br> <li> Introduced functionality to ignore
    specific goroutines during <br>comparison.<br>
    
    </details>
    
      </td>
    <td><a
    href="https://github.com/TykTechnologies/tyk/pull/6812/files#diff-c6edbd39849f1acdcae221ef5e8b6a5886d644ec719064cf051be79f8c9377f8">+124/-0</a>&nbsp;
    </td>
    
    </tr>
    
    <tr>
      <td>
        <details>
    <summary><strong>Dockerfile</strong><dd><code>Simplified Dockerfile with
    Go 1.23 base image.</code>&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;
    &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; </dd></summary>
    <hr>
    
    Dockerfile
    
    <li>Simplified Dockerfile by using Go 1.23-bullseye base image.<br> <li>
    Removed redundant Python installation steps.<br> <li> Optimized build
    process with caching for Go modules.<br>
    
    </details>
    
      </td>
    <td><a
    href="https://github.com/TykTechnologies/tyk/pull/6812/files#diff-dd2c0eb6ea5cfc6c4bd4eac30934e2d5746747af48fef6da689e85b752f39557">+11/-51</a>&nbsp;
    </td>
    
    </tr>
    
    <tr>
      <td>
        <details>
    <summary><strong>Taskfile.yml</strong><dd><code>Enhanced Taskfile to
    support dynamic Go version.</code>&nbsp; &nbsp; &nbsp; &nbsp; &nbsp;
    &nbsp; &nbsp; &nbsp; &nbsp; </dd></summary>
    <hr>
    
    Taskfile.yml
    
    - Added dynamic Go version argument for Docker builds.
    
    </details>
    
      </td>
    <td><a
    href="https://github.com/TykTechnologies/tyk/pull/6812/files#diff-cd2d359855d0301ce190f1ec3b4c572ea690c83747f6df61c9340720e3d2425e">+1/-1</a>&nbsp;
    &nbsp; &nbsp; </td>
    
    </tr>
    
    <tr>
      <td>
        <details>
    <summary><strong>go.mod</strong><dd><code>Bumped Go version in module to
    1.23.4.</code>&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;
    &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;
    &nbsp; </dd></summary>
    <hr>
    
    go.mod
    
    - Updated Go version requirement to 1.23.4.
    
    </details>
    
      </td>
    <td><a
    href="https://github.com/TykTechnologies/tyk/pull/6812/files#diff-33ef32bf6c23acb95f5902d7097b7a1d5128ca061167ec0716715b0b9eeaa5f6">+1/-1</a>&nbsp;
    &nbsp; &nbsp; </td>
    
    </tr>
    </table></details></td></tr><tr><td><strong>Configuration
    changes</strong></td><td><details><summary>4 files</summary><table>
    <tr>
      <td>
        <details>
    <summary><strong>ci-tests.yml</strong><dd><code>Updated CI workflow to
    use Go 1.23.x.</code>&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;
    &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;
    &nbsp; &nbsp; &nbsp; </dd></summary>
    <hr>
    
    .github/workflows/ci-tests.yml
    
    - Updated Go version in CI matrix to 1.23.x.
    
    </details>
    
      </td>
    <td><a
    href="https://github.com/TykTechnologies/tyk/pull/6812/files#diff-03609cb60b0c6e92fb771eb8787d6722b8c31ca4c03eabc788e147acd8c6fb43">+1/-1</a>&nbsp;
    &nbsp; &nbsp; </td>
    
    </tr>
    
    <tr>
      <td>
        <details>
    <summary><strong>plugin-compiler-build.yml</strong><dd><code>Updated
    plugin compiler workflow for Go 1.23.</code>&nbsp; &nbsp; &nbsp; &nbsp;
    &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; </dd></summary>
    <hr>
    
    .github/workflows/plugin-compiler-build.yml
    
    <li>Updated base image to use Go 1.23-bullseye.<br> <li> Fixed
    <code>BASE_IMAGE</code> argument in Docker build steps.<br>
    
    </details>
    
      </td>
    <td><a
    href="https://github.com/TykTechnologies/tyk/pull/6812/files#diff-f3a95a900eb0ac23af6314e9cdea29fa16af0a9bcb61793a83a32ff13d4c4e79">+3/-3</a>&nbsp;
    &nbsp; &nbsp; </td>
    
    </tr>
    
    <tr>
      <td>
        <details>
    <summary><strong>release.yml</strong><dd><code>Updated release workflow
    to support Go 1.23.</code>&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;
    &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; </dd></summary>
    <hr>
    
    .github/workflows/release.yml
    
    <li>Updated Go version in release workflow to 1.23-bullseye.<br> <li>
    Adjusted Docker build conditions for the new Go version.<br>
    
    </details>
    
      </td>
    <td><a
    href="https://github.com/TykTechnologies/tyk/pull/6812/files#diff-87db21a973eed4fef5f32b267aa60fcee5cbdf03c67fafdc2a9b553bb0b15f34">+11/-11</a>&nbsp;
    </td>
    
    </tr>
    
    <tr>
      <td>
        <details>
    <summary><strong>Dockerfile</strong><dd><code>Updated plugin compiler
    Dockerfile for Go 1.23.</code>&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;
    &nbsp; &nbsp; &nbsp; &nbsp; </dd></summary>
    <hr>
    
    ci/images/plugin-compiler/Dockerfile
    
    - Updated base image to Go 1.23-bullseye.
    
    </details>
    
      </td>
    <td><a
    href="https://github.com/TykTechnologies/tyk/pull/6812/files#diff-0ded1ed63ca128bd2d22721b0bc19dc85e440e4922164f465ac647917321971e">+1/-1</a>&nbsp;
    &nbsp; &nbsp; </td>
    
    </tr>
    </table></details></td></tr></tr></tbody></table>
    
    ___
    
    > 💡 **PR-Agent usage**: Comment `/help "your question"` on any pull
    request to receive relevant information
    
    ---------
    
    Co-authored-by: Tit Petric <[email protected]>
    Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
    Projects
    None yet
    Development

    Successfully merging this pull request may close these issues.

    3 participants