Skip to content

Commit

Permalink
support complete insert (pingcap#135)
Browse files Browse the repository at this point in the history
  • Loading branch information
lichunzhu authored Aug 19, 2020
1 parent a85a8a8 commit c27eb2f
Show file tree
Hide file tree
Showing 5 changed files with 26 additions and 11 deletions.
3 changes: 3 additions & 0 deletions dumpling/cmd/dumpling/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,7 @@ var (
csvSeparator string
csvDelimiter string

completeInsert bool
dumpEmptyDatabase bool
escapeBackslash bool
tidbMemQuotaQuery uint64
Expand Down Expand Up @@ -122,6 +123,7 @@ func main() {
pflag.StringVar(&csvSeparator, "csv-separator", ",", "The separator for csv files, default ','")
pflag.StringVar(&csvDelimiter, "csv-delimiter", "\"", "The delimiter for values in csv files, default '\"'")
pflag.StringVar(&outputFilenameFormat, "output-filename-template", "", "The output filename template (without file extension), default '{{.DB}}.{{.Table}}.{{.Index}}'")
pflag.BoolVar(&completeInsert, "complete-insert", false, "Use complete INSERT statements that include column names")

printVersion := pflag.BoolP("version", "V", false, "Print Dumpling version")

Expand Down Expand Up @@ -208,6 +210,7 @@ func main() {
conf.CsvSeparator = csvSeparator
conf.CsvDelimiter = csvDelimiter
conf.OutputFileTemplate = tmpl
conf.CompleteInsert = completeInsert

err = export.Dump(context.Background(), conf)
if err != nil {
Expand Down
1 change: 1 addition & 0 deletions dumpling/v4/export/config.go
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,7 @@ type Config struct {
Rows uint64
Where string
FileType string
CompleteInsert bool
EscapeBackslash bool
DumpEmptyDatabase bool
OutputFileTemplate *template.Template
Expand Down
2 changes: 1 addition & 1 deletion dumpling/v4/export/ir_impl.go
Original file line number Diff line number Diff line change
Expand Up @@ -315,7 +315,7 @@ func splitTableDataIntoChunks(
estimatedStep := (max-min)/estimatedChunks + 1
cutoff := min

selectedField, err := buildSelectField(db, dbName, tableName)
selectedField, err := buildSelectField(db, dbName, tableName, conf.CompleteInsert)
if err != nil {
errCh <- withStack(err)
return
Expand Down
6 changes: 3 additions & 3 deletions dumpling/v4/export/sql.go
Original file line number Diff line number Diff line change
Expand Up @@ -133,7 +133,7 @@ func SelectVersion(db *sql.DB) (string, error) {
}

func SelectAllFromTable(conf *Config, db *sql.Conn, database, table string) (TableDataIR, error) {
selectedField, err := buildSelectField(db, database, table)
selectedField, err := buildSelectField(db, database, table, conf.CompleteInsert)
if err != nil {
return nil, err
}
Expand Down Expand Up @@ -455,7 +455,7 @@ func createConnWithConsistency(ctx context.Context, db *sql.DB) (*sql.Conn, erro
return conn, nil
}

func buildSelectField(db *sql.Conn, dbName, tableName string) (string, error) {
func buildSelectField(db *sql.Conn, dbName, tableName string, completeInsert bool) (string, error) {
query := `SELECT COLUMN_NAME,EXTRA FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_SCHEMA=? AND TABLE_NAME=?;`
rows, err := db.QueryContext(context.Background(), query, dbName, tableName)
if err != nil {
Expand All @@ -479,7 +479,7 @@ func buildSelectField(db *sql.Conn, dbName, tableName string) (string, error) {
}
availableFields = append(availableFields, wrapBackTicks(escapeString(fieldName)))
}
if hasGenerateColumn {
if completeInsert || hasGenerateColumn {
return strings.Join(availableFields, ","), nil
}
return "*", nil
Expand Down
25 changes: 18 additions & 7 deletions dumpling/v4/export/sql_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -75,7 +75,7 @@ func (s *testDumpSuite) TestBuildSelectAllQuery(c *C) {
WithArgs(sqlmock.AnyArg(), sqlmock.AnyArg()).
WillReturnRows(sqlmock.NewRows([]string{"column_name", "extra"}).AddRow("id", ""))

selectedField, err := buildSelectField(conn, "test", "t")
selectedField, err := buildSelectField(conn, "test", "t", false)
c.Assert(err, IsNil)
q := buildSelectQuery("test", "t", selectedField, "", orderByClause)
c.Assert(q, Equals, "SELECT * FROM `test`.`t` ORDER BY _tidb_rowid")
Expand All @@ -91,7 +91,7 @@ func (s *testDumpSuite) TestBuildSelectAllQuery(c *C) {
WithArgs(sqlmock.AnyArg(), sqlmock.AnyArg()).
WillReturnRows(sqlmock.NewRows([]string{"column_name", "extra"}).AddRow("id", ""))

selectedField, err = buildSelectField(conn, "test", "t")
selectedField, err = buildSelectField(conn, "test", "t", false)
c.Assert(err, IsNil)
q = buildSelectQuery("test", "t", selectedField, "", orderByClause)
c.Assert(q, Equals, "SELECT * FROM `test`.`t`")
Expand All @@ -114,7 +114,7 @@ func (s *testDumpSuite) TestBuildSelectAllQuery(c *C) {
WithArgs(sqlmock.AnyArg(), sqlmock.AnyArg()).
WillReturnRows(sqlmock.NewRows([]string{"column_name", "extra"}).AddRow("id", ""))

selectedField, err = buildSelectField(conn, "test", "t")
selectedField, err = buildSelectField(conn, "test", "t", false)
c.Assert(err, IsNil)
q = buildSelectQuery("test", "t", selectedField, "", orderByClause)
c.Assert(q, Equals, "SELECT * FROM `test`.`t` ORDER BY `id`", cmt)
Expand All @@ -138,7 +138,7 @@ func (s *testDumpSuite) TestBuildSelectAllQuery(c *C) {
WithArgs(sqlmock.AnyArg(), sqlmock.AnyArg()).
WillReturnRows(sqlmock.NewRows([]string{"column_name", "extra"}).AddRow("id", ""))

selectedField, err = buildSelectField(conn, "test", "t")
selectedField, err = buildSelectField(conn, "test", "t", false)
c.Assert(err, IsNil)
q := buildSelectQuery("test", "t", selectedField, "", orderByClause)
c.Assert(q, Equals, "SELECT * FROM `test`.`t`", cmt)
Expand All @@ -157,7 +157,7 @@ func (s *testDumpSuite) TestBuildSelectAllQuery(c *C) {
WithArgs(sqlmock.AnyArg(), sqlmock.AnyArg()).
WillReturnRows(sqlmock.NewRows([]string{"column_name", "extra"}).AddRow("id", ""))

selectedField, err := buildSelectField(conn, "test", "t")
selectedField, err := buildSelectField(conn, "test", "t", false)
c.Assert(err, IsNil)
q := buildSelectQuery("test", "t", selectedField, "", "")
c.Assert(q, Equals, "SELECT * FROM `test`.`t`", cmt)
Expand All @@ -177,18 +177,29 @@ func (s *testDumpSuite) TestBuildSelectField(c *C) {
WithArgs(sqlmock.AnyArg(), sqlmock.AnyArg()).
WillReturnRows(sqlmock.NewRows([]string{"column_name", "extra"}).AddRow("id", ""))

selectedField, err := buildSelectField(conn, "test", "t")
selectedField, err := buildSelectField(conn, "test", "t", false)
c.Assert(selectedField, Equals, "*")
c.Assert(err, IsNil)
c.Assert(mock.ExpectationsWereMet(), IsNil)

// user assigns completeInsert
mock.ExpectQuery("SELECT COLUMN_NAME").
WithArgs(sqlmock.AnyArg(), sqlmock.AnyArg()).
WillReturnRows(sqlmock.NewRows([]string{"column_name", "extra"}).AddRow("id", "").
AddRow("name", "").AddRow("quo`te", ""))

selectedField, err = buildSelectField(conn, "test", "t", true)
c.Assert(selectedField, Equals, "`id`,`name`,`quo``te`")
c.Assert(err, IsNil)
c.Assert(mock.ExpectationsWereMet(), IsNil)

// found generate columns, rest columns is `id`,`name`
mock.ExpectQuery("SELECT COLUMN_NAME").
WithArgs(sqlmock.AnyArg(), sqlmock.AnyArg()).
WillReturnRows(sqlmock.NewRows([]string{"column_name", "extra"}).
AddRow("id", "").AddRow("name", "").AddRow("quo`te", "").AddRow("generated", "VIRTUAL GENERATED"))

selectedField, err = buildSelectField(conn, "test", "t")
selectedField, err = buildSelectField(conn, "test", "t", false)
c.Assert(selectedField, Equals, "`id`,`name`,`quo``te`")
c.Assert(err, IsNil)
c.Assert(mock.ExpectationsWereMet(), IsNil)
Expand Down

0 comments on commit c27eb2f

Please sign in to comment.