Skip to content

Commit

Permalink
Merge pull request #8 from adamdecaf/more-nil-panics
Browse files Browse the repository at this point in the history
fix: prevent other nil panics, add CI
  • Loading branch information
sakeven authored Mar 12, 2024
2 parents b3417fa + 7f67782 commit 477e63e
Show file tree
Hide file tree
Showing 4 changed files with 164 additions and 5 deletions.
24 changes: 24 additions & 0 deletions .github/workflows/codeql.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
name: CodeQL Analysis

on:
push:
pull_request:
schedule:
- cron: '0 0 * * 0'

jobs:
CodeQL-Build:
strategy:
fail-fast: false
runs-on: ubuntu-latest
steps:
- name: Checkout repository
uses: actions/checkout@v2

- name: Initialize CodeQL
uses: github/codeql-action/init@v2
with:
languages: go

- name: Perform CodeQL Analysis
uses: github/codeql-action/analyze@v2
32 changes: 32 additions & 0 deletions .github/workflows/go.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
name: Go

on:
push:
branches: [ master ]
pull_request:
branches: [ master ]

jobs:
build:
name: Go Build
runs-on: ${{ matrix.os }}
strategy:
matrix:
os: [ubuntu-latest, macos-latest, windows-latest]
steps:
- name: Set up Go 1.x
uses: actions/setup-go@v4
with:
go-version: stable
id: go

- name: Check out code into the Go module directory
uses: actions/checkout@v2

- name: Go vet
run: |
go vet ./...
- name: Go Test
run: |
go test ./...
23 changes: 19 additions & 4 deletions rbtree.go
Original file line number Diff line number Diff line change
Expand Up @@ -30,22 +30,28 @@ func NewTree[K Ordered, V any]() *Tree[K, V] {

// Find finds the node and return its value.
func (t *Tree[K, V]) Find(key K) V {
var result V
if t == nil {
return result
}
n := t.findnode(key)
if n != nil {
return n.Value
}
var result V
return result
}

// FindIt finds the node and return it as an iterator.
func (t *Tree[K, V]) FindIt(key K) *node[K, V] {
if t == nil || t.root == nil {
return nil
}
return t.findnode(key)
}

// Empty checks whether the rbtree is empty.
func (t *Tree[K, V]) Empty() bool {
if t.root == nil {
if t == nil || t.root == nil {
return true
}
return false
Expand All @@ -61,13 +67,18 @@ func (t *Tree[K, V]) Iterator() *node[K, V] {

// Size returns the size of the rbtree.
func (t *Tree[K, V]) Size() int {
if t == nil {
return 0
}
return t.size
}

// Clear destroys the rbtree.
func (t *Tree[K, V]) Clear() {
t.root = nil
t.size = 0
if t != nil {
t.root = nil
t.size = 0
}
}

// Insert inserts the key-value pair into the rbtree.
Expand Down Expand Up @@ -102,6 +113,10 @@ func (t *Tree[K, V]) Insert(key K, value V) {

// Delete deletes the node by key
func (t *Tree[K, V]) Delete(key K) {
if t == nil || t.root == nil {
return
}

z := t.findnode(key)
if z == nil {
return
Expand Down
90 changes: 89 additions & 1 deletion rbtree_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ package rbtree

import (
"fmt"
"strings"
"testing"
)

Expand Down Expand Up @@ -57,6 +58,56 @@ func TestPreorder(t *testing.T) {
tree.Preorder()
}

func TestTree(t *testing.T) {
tree := NewTree[int, string]()

t.Run("empty", func(t *testing.T) {
tree.Clear()
tree.Delete(1)

if !tree.Empty() {
t.Fatal("tree isn't empty")
}

size := tree.Size()
if size != 0 {
t.Errorf("unexpected tree size of %d", size)
}
})

t.Run("nil", func(t *testing.T) {
tree = nil

tree.Clear()
tree.Delete(1)

if !tree.Empty() {
t.Fatal("tree isn't empty")
}

size := tree.Size()
if size != 0 {
t.Errorf("unexpected tree size of %d", size)
}
})

var caught interface{}
t.Run("catch insert panic", func(t *testing.T) {
defer func() {
if err := recover(); err != nil {
caught = err
}
}()
// panics
tree.Insert(1, "abc")
})

error := fmt.Sprintf("%v", caught)
if !strings.Contains(error, "nil pointer dereference") {
t.Fatalf("unexpected error: %#v", caught)
}
}

func TestFind(t *testing.T) {
tree := NewTree[int, string]()

Expand Down Expand Up @@ -93,6 +144,20 @@ func TestFind(t *testing.T) {
t.Fatalf("got %q", value)
}
})

t.Run("nil", func(t *testing.T) {
tree = nil

n := tree.FindIt(4)
if n != nil {
t.Fatalf("got %#v", n)
}

value := tree.Find(5)
if value != "" {
t.Fatalf("got %q", value)
}
})
}

func TestIterator(t *testing.T) {
Expand All @@ -115,7 +180,20 @@ func TestIterator(t *testing.T) {
tree = NewTree[int, string]()

next := tree.Iterator()
t.Logf("tree.Iterator()=%#v", next)
if next != nil {
t.Fatalf(".Iterator() returned %#v", next)
}

size := tree.Size()
if size != 0 {
t.Fatalf("got size %d", size)
}
})

t.Run("nil", func(t *testing.T) {
tree = nil

next := tree.Iterator()
if next != nil {
t.Fatalf(".Iterator() returned %#v", next)
}
Expand Down Expand Up @@ -158,6 +236,16 @@ func TestDelete(t *testing.T) {
t.Fatalf("after size is %d", size)
}
})

t.Run("nil", func(t *testing.T) {
tree = nil
tree.Delete(1)

size := tree.Size()
if size != 0 {
t.Fatalf("after size is %d", size)
}
})
}

func TestDelete2(t *testing.T) {
Expand Down

0 comments on commit 477e63e

Please sign in to comment.