-
Notifications
You must be signed in to change notification settings - Fork 2.1k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add cycle detection for foreign keys (#14339)
Signed-off-by: Manan Gupta <manan@planetscale.com>
- Loading branch information
1 parent
563c921
commit e802cff
Showing
17 changed files
with
544 additions
and
12 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,119 @@ | ||
/* | ||
Copyright 2023 The Vitess Authors. | ||
Licensed under the Apache License, Version 2.0 (the "License"); | ||
you may not use this file except in compliance with the License. | ||
You may obtain a copy of the License at | ||
http://www.apache.org/licenses/LICENSE-2.0 | ||
Unless required by applicable law or agreed to in writing, software | ||
distributed under the License is distributed on an "AS IS" BASIS, | ||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
See the License for the specific language governing permissions and | ||
limitations under the License. | ||
*/ | ||
|
||
package graph | ||
|
||
import ( | ||
"fmt" | ||
"slices" | ||
"strings" | ||
) | ||
|
||
// Graph is a generic graph implementation. | ||
type Graph[C comparable] struct { | ||
edges map[C][]C | ||
} | ||
|
||
// NewGraph creates a new graph for the given comparable type. | ||
func NewGraph[C comparable]() *Graph[C] { | ||
return &Graph[C]{ | ||
edges: map[C][]C{}, | ||
} | ||
} | ||
|
||
// AddVertex adds a vertex to the given Graph. | ||
func (gr *Graph[C]) AddVertex(vertex C) { | ||
_, alreadyExists := gr.edges[vertex] | ||
if alreadyExists { | ||
return | ||
} | ||
gr.edges[vertex] = []C{} | ||
} | ||
|
||
// AddEdge adds an edge to the given Graph. | ||
// It also makes sure that the vertices are added to the graph if not already present. | ||
func (gr *Graph[C]) AddEdge(start C, end C) { | ||
gr.AddVertex(start) | ||
gr.AddVertex(end) | ||
gr.edges[start] = append(gr.edges[start], end) | ||
} | ||
|
||
// PrintGraph is used to print the graph. This is only used for testing. | ||
func (gr *Graph[C]) PrintGraph() string { | ||
adjacencyLists := []string{} | ||
for vertex, edges := range gr.edges { | ||
adjacencyList := fmt.Sprintf("%v -", vertex) | ||
for _, end := range edges { | ||
adjacencyList += fmt.Sprintf(" %v", end) | ||
} | ||
adjacencyLists = append(adjacencyLists, adjacencyList) | ||
} | ||
slices.Sort(adjacencyLists) | ||
return strings.Join(adjacencyLists, "\n") | ||
} | ||
|
||
// Empty checks whether the graph is empty or not. | ||
func (gr *Graph[C]) Empty() bool { | ||
return len(gr.edges) == 0 | ||
} | ||
|
||
// HasCycles checks whether the given graph has a cycle or not. | ||
// We are using a well-known DFS based colouring algorithm to check for cycles. | ||
// Look at https://cp-algorithms.com/graph/finding-cycle.html for more details on the algorithm. | ||
func (gr *Graph[C]) HasCycles() bool { | ||
// If the graph is empty, then we don't need to check anything. | ||
if gr.Empty() { | ||
return false | ||
} | ||
// Initialize the coloring map. | ||
// 0 represents white. | ||
// 1 represents grey. | ||
// 2 represents black. | ||
color := map[C]int{} | ||
for vertex := range gr.edges { | ||
// If any vertex is still white, we initiate a new DFS. | ||
if color[vertex] == 0 { | ||
if gr.hasCyclesDfs(color, vertex) { | ||
return true | ||
} | ||
} | ||
} | ||
return false | ||
} | ||
|
||
// hasCyclesDfs is a utility function for checking for cycles in a graph. | ||
// It runs a dfs from the given vertex marking each vertex as grey. During the dfs, | ||
// if we encounter a grey vertex, we know we have a cycle. We mark the visited vertices black | ||
// on finishing the dfs. | ||
func (gr *Graph[C]) hasCyclesDfs(color map[C]int, vertex C) bool { | ||
// Mark the vertex grey. | ||
color[vertex] = 1 | ||
// Go over all the edges. | ||
for _, end := range gr.edges[vertex] { | ||
// If we encounter a white vertex, we continue the dfs. | ||
if color[end] == 0 { | ||
if gr.hasCyclesDfs(color, end) { | ||
return true | ||
} | ||
} else if color[end] == 1 { | ||
// We encountered a grey vertex, we have a cycle. | ||
return true | ||
} | ||
} | ||
// Mark the vertex black before finishing | ||
color[vertex] = 2 | ||
return false | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,153 @@ | ||
/* | ||
Copyright 2023 The Vitess Authors. | ||
Licensed under the Apache License, Version 2.0 (the "License"); | ||
you may not use this file except in compliance with the License. | ||
You may obtain a copy of the License at | ||
http://www.apache.org/licenses/LICENSE-2.0 | ||
Unless required by applicable law or agreed to in writing, software | ||
distributed under the License is distributed on an "AS IS" BASIS, | ||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
See the License for the specific language governing permissions and | ||
limitations under the License. | ||
*/ | ||
|
||
package graph | ||
|
||
import ( | ||
"testing" | ||
|
||
"github.com/stretchr/testify/require" | ||
) | ||
|
||
// TestIntegerGraph tests that a graph with Integers can be created and all graph functions work as intended. | ||
func TestIntegerGraph(t *testing.T) { | ||
testcases := []struct { | ||
name string | ||
edges [][2]int | ||
wantedGraph string | ||
wantEmpty bool | ||
wantHasCycles bool | ||
}{ | ||
{ | ||
name: "empty graph", | ||
edges: nil, | ||
wantedGraph: "", | ||
wantEmpty: true, | ||
wantHasCycles: false, | ||
}, { | ||
name: "non-cyclic graph", | ||
edges: [][2]int{ | ||
{1, 2}, | ||
{2, 3}, | ||
{1, 4}, | ||
{2, 5}, | ||
{4, 5}, | ||
}, | ||
wantedGraph: `1 - 2 4 | ||
2 - 3 5 | ||
3 - | ||
4 - 5 | ||
5 -`, | ||
wantEmpty: false, | ||
wantHasCycles: false, | ||
}, { | ||
name: "cyclic graph", | ||
edges: [][2]int{ | ||
{1, 2}, | ||
{2, 3}, | ||
{1, 4}, | ||
{2, 5}, | ||
{4, 5}, | ||
{5, 6}, | ||
{6, 1}, | ||
}, | ||
wantedGraph: `1 - 2 4 | ||
2 - 3 5 | ||
3 - | ||
4 - 5 | ||
5 - 6 | ||
6 - 1`, | ||
wantEmpty: false, | ||
wantHasCycles: true, | ||
}, | ||
} | ||
for _, tt := range testcases { | ||
t.Run(tt.name, func(t *testing.T) { | ||
graph := NewGraph[int]() | ||
for _, edge := range tt.edges { | ||
graph.AddEdge(edge[0], edge[1]) | ||
} | ||
require.Equal(t, tt.wantedGraph, graph.PrintGraph()) | ||
require.Equal(t, tt.wantEmpty, graph.Empty()) | ||
require.Equal(t, tt.wantHasCycles, graph.HasCycles()) | ||
}) | ||
} | ||
} | ||
|
||
// TestStringGraph tests that a graph with strings can be created and all graph functions work as intended. | ||
func TestStringGraph(t *testing.T) { | ||
testcases := []struct { | ||
name string | ||
edges [][2]string | ||
wantedGraph string | ||
wantEmpty bool | ||
wantHasCycles bool | ||
}{ | ||
{ | ||
name: "empty graph", | ||
edges: nil, | ||
wantedGraph: "", | ||
wantEmpty: true, | ||
wantHasCycles: false, | ||
}, { | ||
name: "non-cyclic graph", | ||
edges: [][2]string{ | ||
{"A", "B"}, | ||
{"B", "C"}, | ||
{"A", "D"}, | ||
{"B", "E"}, | ||
{"D", "E"}, | ||
}, | ||
wantedGraph: `A - B D | ||
B - C E | ||
C - | ||
D - E | ||
E -`, | ||
wantEmpty: false, | ||
wantHasCycles: false, | ||
}, { | ||
name: "cyclic graph", | ||
edges: [][2]string{ | ||
{"A", "B"}, | ||
{"B", "C"}, | ||
{"A", "D"}, | ||
{"B", "E"}, | ||
{"D", "E"}, | ||
{"E", "F"}, | ||
{"F", "A"}, | ||
}, | ||
wantedGraph: `A - B D | ||
B - C E | ||
C - | ||
D - E | ||
E - F | ||
F - A`, | ||
wantEmpty: false, | ||
wantHasCycles: true, | ||
}, | ||
} | ||
for _, tt := range testcases { | ||
t.Run(tt.name, func(t *testing.T) { | ||
graph := NewGraph[string]() | ||
for _, edge := range tt.edges { | ||
graph.AddEdge(edge[0], edge[1]) | ||
} | ||
require.Equal(t, tt.wantedGraph, graph.PrintGraph()) | ||
require.Equal(t, tt.wantEmpty, graph.Empty()) | ||
require.Equal(t, tt.wantHasCycles, graph.HasCycles()) | ||
}) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.