// Copyright 2016 PingCAP, Inc. // // 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, // See the License for the specific language governing permissions and // limitations under the License. package ddl import ( "strings" "sync" "github.com/hanchuanchuan/goInception/ast" "github.com/hanchuanchuan/goInception/kv" "github.com/hanchuanchuan/goInception/model" "github.com/hanchuanchuan/goInception/sessionctx" "github.com/hanchuanchuan/goInception/table" "github.com/hanchuanchuan/goInception/util/testleak" . "github.com/pingcap/check" "github.com/pingcap/errors" "golang.org/x/net/context" ) var _ = Suite(&testForeighKeySuite{}) type testForeighKeySuite struct { store kv.Storage dbInfo *model.DBInfo d *ddl ctx sessionctx.Context } func (s *testForeighKeySuite) SetUpSuite(c *C) { testleak.BeforeTest() s.store = testCreateStore(c, "test_foreign") } func (s *testForeighKeySuite) TearDownSuite(c *C) { err := s.store.Close() c.Assert(err, IsNil) testleak.AfterTest(c)() } func (s *testForeighKeySuite) testCreateForeignKey(c *C, tblInfo *model.TableInfo, fkName string, keys []string, refTable string, refKeys []string, onDelete ast.ReferOptionType, onUpdate ast.ReferOptionType) *model.Job { FKName := model.NewCIStr(fkName) Keys := make([]model.CIStr, len(keys)) for i, key := range keys { Keys[i] = model.NewCIStr(key) } RefTable := model.NewCIStr(refTable) RefKeys := make([]model.CIStr, len(refKeys)) for i, key := range refKeys { RefKeys[i] = model.NewCIStr(key) } fkInfo := &model.FKInfo{ Name: FKName, RefTable: RefTable, RefCols: RefKeys, Cols: Keys, OnDelete: int(onDelete), OnUpdate: int(onUpdate), State: model.StateNone, } job := &model.Job{ SchemaID: s.dbInfo.ID, TableID: tblInfo.ID, Type: model.ActionAddForeignKey, BinlogInfo: &model.HistoryInfo{}, Args: []interface{}{fkInfo}, } err := s.d.doDDLJob(s.ctx, job) c.Assert(err, IsNil) return job } func testDropForeignKey(c *C, ctx sessionctx.Context, d *ddl, dbInfo *model.DBInfo, tblInfo *model.TableInfo, foreignKeyName string) *model.Job { job := &model.Job{ SchemaID: dbInfo.ID, TableID: tblInfo.ID, Type: model.ActionDropForeignKey, BinlogInfo: &model.HistoryInfo{}, Args: []interface{}{model.NewCIStr(foreignKeyName)}, } err := d.doDDLJob(ctx, job) c.Assert(err, IsNil) v := getSchemaVer(c, ctx) checkHistoryJobArgs(c, ctx, job.ID, &historyJobArgs{ver: v, tbl: tblInfo}) return job } func getForeignKey(t table.Table, name string) *model.FKInfo { for _, fk := range t.Meta().ForeignKeys { // only public foreign key can be read. if fk.State != model.StatePublic { continue } if fk.Name.L == strings.ToLower(name) { return fk } } return nil } func (s *testForeighKeySuite) TestForeignKey(c *C) { d := testNewDDL(context.Background(), nil, s.store, nil, nil, testLease) defer d.Stop() s.d = d s.dbInfo = testSchemaInfo(c, d, "test_foreign") ctx := testNewContext(d) s.ctx = ctx testCreateSchema(c, ctx, d, s.dbInfo) tblInfo := testTableInfo(c, d, "t", 3) err := ctx.NewTxn() c.Assert(err, IsNil) testCreateTable(c, ctx, d, s.dbInfo, tblInfo) err = ctx.Txn().Commit(context.Background()) c.Assert(err, IsNil) // fix data race var mu sync.Mutex checkOK := false var hookErr error tc := &TestDDLCallback{} tc.onJobUpdated = func(job *model.Job) { if job.State != model.JobStateDone { return } mu.Lock() defer mu.Unlock() var t table.Table t, err = testGetTableWithError(d, s.dbInfo.ID, tblInfo.ID) if err != nil { hookErr = errors.Trace(err) return } fk := getForeignKey(t, "c1_fk") if fk == nil { hookErr = errors.New("foreign key not exists") return } checkOK = true } d.SetHook(tc) d.Stop() d.start(context.Background(), nil) job := s.testCreateForeignKey(c, tblInfo, "c1_fk", []string{"c1"}, "t2", []string{"c1"}, ast.ReferOptionCascade, ast.ReferOptionSetNull) testCheckJobDone(c, d, job, true) err = ctx.Txn().Commit(context.Background()) c.Assert(err, IsNil) mu.Lock() hErr := hookErr ok := checkOK mu.Unlock() c.Assert(hErr, IsNil) c.Assert(ok, IsTrue) v := getSchemaVer(c, ctx) checkHistoryJobArgs(c, ctx, job.ID, &historyJobArgs{ver: v, tbl: tblInfo}) mu.Lock() checkOK = false mu.Unlock() tc.onJobUpdated = func(job *model.Job) { if job.State != model.JobStateDone { return } mu.Lock() defer mu.Unlock() var t table.Table t, err = testGetTableWithError(d, s.dbInfo.ID, tblInfo.ID) if err != nil { hookErr = errors.Trace(err) return } fk := getForeignKey(t, "c1_fk") if fk != nil { hookErr = errors.New("foreign key has not been dropped") return } checkOK = true } d.Stop() d.start(context.Background(), nil) job = testDropForeignKey(c, ctx, d, s.dbInfo, tblInfo, "c1_fk") testCheckJobDone(c, d, job, false) mu.Lock() hErr = hookErr ok = checkOK mu.Unlock() c.Assert(hErr, IsNil) c.Assert(ok, IsTrue) err = ctx.NewTxn() c.Assert(err, IsNil) tc.onJobUpdated = func(job *model.Job) { } d.Stop() d.start(context.Background(), nil) job = testDropTable(c, ctx, d, s.dbInfo, tblInfo) testCheckJobDone(c, d, job, false) err = ctx.Txn().Commit(context.Background()) c.Assert(err, IsNil) }