Skip to content

Commit

Permalink
Test added for verification api
Browse files Browse the repository at this point in the history
  • Loading branch information
Vivek Yadav authored and akashthawaitcc committed Dec 26, 2024
1 parent 66aebbe commit bbee059
Show file tree
Hide file tree
Showing 3 changed files with 120 additions and 61 deletions.
10 changes: 10 additions & 0 deletions sources/common/toddl.go
Original file line number Diff line number Diff line change
Expand Up @@ -158,6 +158,16 @@ func (ss *SchemaToSpannerImpl) VerifyExpressions(conv *internal.Conv) {
return
}

if conv.SchemaIssues == nil {
conv.SchemaIssues = make(map[string]internal.TableIssues)
}

if _, exists := conv.SchemaIssues[tableId]; !exists {
conv.SchemaIssues[tableId] = internal.TableIssues{
ColumnLevelIssues: make(map[string][]internal.SchemaIssue),
}
}

colIssue := conv.SchemaIssues[tableId].ColumnLevelIssues[colId]

if !IsSchemaIssuePresent(colIssue, issueType) {
Expand Down
10 changes: 10 additions & 0 deletions webv2/api/schema.go
Original file line number Diff line number Diff line change
Expand Up @@ -618,6 +618,16 @@ func (expressionVerificationHandler *ExpressionsVerificationHandler) VerifyCheck
return
}

if sessionState.Conv.SchemaIssues == nil {
sessionState.Conv.SchemaIssues = make(map[string]internal.TableIssues)
}

if _, exists := sessionState.Conv.SchemaIssues[tableId]; !exists {
sessionState.Conv.SchemaIssues[tableId] = internal.TableIssues{
ColumnLevelIssues: make(map[string][]internal.SchemaIssue),
}
}

colIssue := sessionState.Conv.SchemaIssues[tableId].ColumnLevelIssues[colId]

if !utilities.IsSchemaIssuePresent(colIssue, issueType) {
Expand Down
161 changes: 100 additions & 61 deletions webv2/api/schema_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,9 @@ package api_test
import (
"bytes"
"encoding/json"
"errors"
"fmt"
"io"
"io/ioutil"
"net/http"
"net/http/httptest"
"strings"
Expand Down Expand Up @@ -1380,7 +1380,7 @@ func TestRestoreSecondaryIndex(t *testing.T) {
"t1": {
Name: "table1",
ColIds: []string{"c1", "c2", "c3"},
Indexes: []ddl.CreateIndex{
Indexes: []ddl.CreateIndex{
{Name: "idx2", TableId: "t1", Unique: false, Keys: []ddl.IndexKey{{ColId: "c3", Desc: false, Order: 1}}, Id: "i2"},
},
Id: "t1",
Expand Down Expand Up @@ -2629,66 +2629,105 @@ func (errReader) Read(p []byte) (n int, err error) {
return 0, fmt.Errorf("simulated read error")
}

func TestVerifyCheckConstraintExpression(t *testing.T) {
// Arrange
mockAccessor := new(mocks.MockExpressionVerificationAccessor)
handler := &api.ExpressionsVerificationHandler{ExpressionVerificationAccessor: mockAccessor}

req, err := http.NewRequest("POST", "/checkConstraint", nil) // Set nil as we'll overwrite it
if err != nil {
t.Fatal(err)
}

// Simulate context and setup session
ctx := req.Context()
sessionState := session.GetSessionState()
sessionState.Driver = constants.MYSQL
sessionState.SpannerInstanceID = "foo"
sessionState.SpannerProjectId = "daring-12"
sessionState.Conv = internal.MakeConv()
sessionState.Conv.SpSchema = map[string]ddl.CreateTable{
"t1": {
Name: "table1",
Id: "t1",
PrimaryKeys: []ddl.IndexKey{{ColId: "c1"}},
CheckConstraints: []ddl.CheckConstraint{
{Expr: "col1 > 0", ExprId: "expr1", Name: "check1"},
func TestVerifyCheckConstraintExpressions(t *testing.T) {
tests := []struct {
name string
expressions []ddl.CheckConstraint
expectedResults []internal.ExpressionVerificationOutput
expectedResponse bool
}{
{
name: "AllValidExpressions",
expressions: []ddl.CheckConstraint{
{Expr: "(col1 > 0)", ExprId: "expr1", Name: "check1"},
},
expectedResults: []internal.ExpressionVerificationOutput{
{Result: true, Err: nil, ExpressionDetail: internal.ExpressionDetail{Expression: "(col1 > 0)", Type: "CHECK", Metadata: map[string]string{"tableId": "t1", "colId": "c1", "checkConstraintName": "check1"}, ExpressionId: "expr1"}},
},
expectedResponse: false,
},
{
name: "InvalidSyntaxError",
expressions: []ddl.CheckConstraint{
{Expr: "(col1 > 0)", ExprId: "expr1", Name: "check1"},
{Expr: "(col1 > 18", ExprId: "expr2", Name: "check2"},
},
expectedResults: []internal.ExpressionVerificationOutput{
{Result: true, Err: nil, ExpressionDetail: internal.ExpressionDetail{Expression: "(col1 > 0)", Type: "CHECK", Metadata: map[string]string{"tableId": "t1", "colId": "c1", "checkConstraintName": "check1"}, ExpressionId: "expr1"}},
{Result: false, Err: errors.New("Syntax error ..."), ExpressionDetail: internal.ExpressionDetail{Expression: "(col1 > 18", Type: "CHECK", Metadata: map[string]string{"tableId": "t1", "colId": "c1", "checkConstraintName": "check2"}, ExpressionId: "expr2"}},
},
expectedResponse: true,
},
{
name: "NameError",
expressions: []ddl.CheckConstraint{
{Expr: "(col1 > 0)", ExprId: "expr1", Name: "check1"},
{Expr: "(col1 > 18)", ExprId: "expr2", Name: "check2"},
},
expectedResults: []internal.ExpressionVerificationOutput{
{Result: true, Err: nil, ExpressionDetail: internal.ExpressionDetail{Expression: "(col1 > 0)", Type: "CHECK", Metadata: map[string]string{"tableId": "t1", "colId": "c1", "checkConstraintName": "check1"}, ExpressionId: "expr1"}},
{Result: false, Err: errors.New("Unrecognized name ..."), ExpressionDetail: internal.ExpressionDetail{Expression: "(col1 > 18)", Type: "CHECK", Metadata: map[string]string{"tableId": "t1", "colId": "c1", "checkConstraintName": "check2"}, ExpressionId: "expr2"}},
},
expectedResponse: true,
},
{
name: "TypeError",
expressions: []ddl.CheckConstraint{
{Expr: "(col1 > 0)", ExprId: "expr1", Name: "check1"},
{Expr: "(col1 > 18)", ExprId: "expr2", Name: "check2"},
},
expectedResults: []internal.ExpressionVerificationOutput{
{Result: true, Err: nil, ExpressionDetail: internal.ExpressionDetail{Expression: "(col1 > 0)", Type: "CHECK", Metadata: map[string]string{"tableId": "t1", "colId": "c1", "checkConstraintName": "check1"}, ExpressionId: "expr1"}},
{Result: false, Err: errors.New("No matching signature for operator"), ExpressionDetail: internal.ExpressionDetail{Expression: "(col1 > 18)", Type: "CHECK", Metadata: map[string]string{"tableId": "t1", "colId": "c1", "checkConstraintName": "check2"}, ExpressionId: "expr2"}},
},
expectedResponse: true,
},
}

// Setup request body
expressionDetails := []internal.ExpressionDetail{
// Add expression details to verify
}
body, _ := json.Marshal(expressionDetails)
req.Body = ioutil.NopCloser(bytes.NewReader(body))

// Mock VerifyExpressions
mockAccessor.On("VerifyExpressions", ctx, mock.Anything).Return(internal.VerifyExpressionsOutput{
ExpressionVerificationOutputList: []internal.ExpressionVerificationOutput{
{
Result: true,
},
},
})

// Use httptest to record the response
rr := httptest.NewRecorder()

// Act
handler.VerifyCheckConstraintExpression(rr, req)

// Assert
assert.Equal(t, http.StatusOK, rr.Code)

var response bool
err = json.NewDecoder(rr.Body).Decode(&response)
if err != nil {
t.Fatal(err)
}
assert.False(t, response) // No errors should have occurred if no invalid expressions

// Verify if the VerifyExpressions was called
mockAccessor.AssertNumberOfCalls(t, "VerifyExpressions", 1)
}
for _, tc := range tests {
t.Run(tc.name, func(t *testing.T) {
mockAccessor := new(mocks.MockExpressionVerificationAccessor)
handler := &api.ExpressionsVerificationHandler{ExpressionVerificationAccessor: mockAccessor}

req, err := http.NewRequest("POST", "/verifyCheckConstraintExpression", nil)
if err != nil {
t.Fatal(err)
}

ctx := req.Context()
sessionState := session.GetSessionState()
sessionState.Driver = constants.MYSQL
sessionState.SpannerInstanceID = "foo"
sessionState.SpannerProjectId = "daring-12"
sessionState.Conv = internal.MakeConv()
sessionState.Conv.SpSchema = map[string]ddl.CreateTable{
"t1": {
Name: "table1",
Id: "t1",
PrimaryKeys: []ddl.IndexKey{{ColId: "c1"}},
ColIds: []string{"c1"},
ColDefs: map[string]ddl.ColumnDef{
"c1": {Name: "col1", Id: "c1", T: ddl.Type{Name: ddl.Int64}},
},
CheckConstraints: tc.expressions,
},
}

mockAccessor.On("VerifyExpressions", ctx, mock.Anything).Return(internal.VerifyExpressionsOutput{
ExpressionVerificationOutputList: tc.expectedResults,
})

rr := httptest.NewRecorder()
handler.VerifyCheckConstraintExpression(rr, req)

assert.Equal(t, http.StatusOK, rr.Code)

var response bool
err = json.NewDecoder(rr.Body).Decode(&response)
if err != nil {
t.Fatal(err)
}
assert.Equal(t, tc.expectedResponse, response)
})
}
}

0 comments on commit bbee059

Please sign in to comment.